// // DownloadTests.swift // // Copyright (c) 2014-2016 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 class DownloadInitializationTestCase: BaseTestCase { let searchPathDirectory: FileManager.SearchPathDirectory = .cachesDirectory let searchPathDomain: FileManager.SearchPathDomainMask = .userDomainMask func testDownloadClassMethodWithMethodURLAndDestination() { // Given let urlString = "https://httpbin.org/" let destination = DownloadRequest.suggestedDownloadDestination(for: searchPathDirectory, in: searchPathDomain) // When let request = Alamofire.download(urlString, to: destination, withMethod: .get) // Then XCTAssertNotNil(request.request) XCTAssertEqual(request.request?.httpMethod, "GET") XCTAssertEqual(request.request?.urlString, urlString) XCTAssertNil(request.response) } func testDownloadClassMethodWithMethodURLHeadersAndDestination() { // Given let urlString = "https://httpbin.org/" let headers = ["Authorization": "123456"] let destination = DownloadRequest.suggestedDownloadDestination(for: searchPathDirectory, in: searchPathDomain) // When let request = Alamofire.download(urlString, to: destination, withMethod: .get, headers: headers) // Then XCTAssertNotNil(request.request) XCTAssertEqual(request.request?.httpMethod, "GET") XCTAssertEqual(request.request?.urlString, urlString) XCTAssertEqual(request.request?.value(forHTTPHeaderField: "Authorization"), "123456") XCTAssertNil(request.response) } } // MARK: - class DownloadResponseTestCase: BaseTestCase { let searchPathDirectory: FileManager.SearchPathDirectory = .cachesDirectory let searchPathDomain: FileManager.SearchPathDomainMask = .userDomainMask let cachesURL: URL = { let cachesDirectory = NSSearchPathForDirectoriesInDomains(.cachesDirectory, .userDomainMask, true).first! let cachesURL = URL(fileURLWithPath: cachesDirectory, isDirectory: true) return cachesURL }() var randomCachesFileURL: URL { return cachesURL.appendingPathComponent("\(UUID().uuidString).json") } func testDownloadRequest() { // Given let numberOfLines = 100 let urlString = "https://httpbin.org/stream/\(numberOfLines)" let destination = DownloadRequest.suggestedDownloadDestination(for: searchPathDirectory, in: searchPathDomain) let expectation = self.expectation(description: "Download request should download data to file: \(urlString)") var response: DefaultDownloadResponse? // When Alamofire.download(urlString, to: destination, withMethod: .get) .response { resp in response = resp expectation.fulfill() } waitForExpectations(timeout: timeout, handler: nil) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNotNil(response?.destinationURL) XCTAssertNil(response?.resumeData) XCTAssertNil(response?.error) if let destinationURL = response?.destinationURL { 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 testDownloadRequestWithProgress() { // Given let randomBytes = 4 * 1024 * 1024 let urlString = "https://httpbin.org/bytes/\(randomBytes)" let fileManager = FileManager.default let directory = fileManager.urls(for: searchPathDirectory, in: self.searchPathDomain)[0] let filename = "test_download_data" let fileURL = directory.appendingPathComponent(filename) let expectation = self.expectation(description: "Bytes download progress should be reported: \(urlString)") var byteValues: [(bytes: Int64, totalBytes: Int64, totalBytesExpected: Int64)] = [] var progressValues: [(completedUnitCount: Int64, totalUnitCount: Int64)] = [] var response: DefaultDownloadResponse? // When Alamofire.download(urlString, to: { _, _ in fileURL }, withMethod: .get) .downloadProgress { progress in progressValues.append((progress.completedUnitCount, progress.totalUnitCount)) } .downloadProgress { bytesRead, totalBytesRead, totalBytesExpectedToRead in let bytes = (bytes: bytesRead, totalBytes: totalBytesRead, totalBytesExpected: totalBytesExpectedToRead) byteValues.append(bytes) } .response { resp in response = resp expectation.fulfill() } waitForExpectations(timeout: timeout, handler: nil) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNotNil(response?.destinationURL) XCTAssertNil(response?.resumeData) XCTAssertNil(response?.error) XCTAssertEqual(byteValues.count, progressValues.count) if byteValues.count == progressValues.count { for (byteValue, progressValue) in zip(byteValues, progressValues) { XCTAssertGreaterThan(byteValue.bytes, 0) print("\(byteValue.totalBytes) - \(progressValue.completedUnitCount)") XCTAssertEqual(byteValue.totalBytes, progressValue.completedUnitCount) XCTAssertEqual(byteValue.totalBytesExpected, progressValue.totalUnitCount) } } if let lastByteValue = byteValues.last, let lastProgressValue = progressValues.last { let byteValueFractionalCompletion = Double(lastByteValue.totalBytes) / Double(lastByteValue.totalBytesExpected) let progressValueFractionalCompletion = Double(lastProgressValue.0) / Double(lastProgressValue.1) XCTAssertEqual(byteValueFractionalCompletion, 1.0) XCTAssertEqual(progressValueFractionalCompletion, 1.0) } else { XCTFail("last item in bytesValues and progressValues should not be nil") } do { try fileManager.removeItem(at: fileURL) } catch { XCTFail("file manager should remove item at URL: \(fileURL)") } } func testDownloadRequestWithParameters() { // Given let fileURL = randomCachesFileURL let urlString = "https://httpbin.org/get" let parameters = ["foo": "bar"] let destination: DownloadRequest.DownloadFileDestination = { _, _ in fileURL } let expectation = self.expectation(description: "Download request should download data to file: \(fileURL)") var response: DefaultDownloadResponse? // When Alamofire.download(urlString, to: destination, withMethod: .get, parameters: parameters) .response { resp in response = resp expectation.fulfill() } waitForExpectations(timeout: timeout, handler: nil) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNotNil(response?.destinationURL) XCTAssertNil(response?.resumeData) XCTAssertNil(response?.error) if let data = try? Data(contentsOf: fileURL), let jsonObject = try? JSONSerialization.jsonObject(with: data, options: JSONSerialization.ReadingOptions(rawValue: 0)), 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 urlString = "https://httpbin.org/get" let headers = ["Authorization": "123456"] let destination: DownloadRequest.DownloadFileDestination = { _, _ in fileURL } let expectation = self.expectation(description: "Download request should download data to file: \(fileURL)") var response: DefaultDownloadResponse? // When Alamofire.download(urlString, to: destination, withMethod: .get, headers: headers) .response { resp in response = resp expectation.fulfill() } waitForExpectations(timeout: timeout, handler: nil) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNotNil(response?.destinationURL) XCTAssertNil(response?.resumeData) XCTAssertNil(response?.error) if let data = try? Data(contentsOf: fileURL), let jsonObject = try? JSONSerialization.jsonObject(with: data, options: JSONSerialization.ReadingOptions(rawValue: 0)), 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") } } } // MARK: - class DownloadResumeDataTestCase: BaseTestCase { let urlString = "https://upload.wikimedia.org/wikipedia/commons/6/69/NASA-HS201427a-HubbleUltraDeepField2014-20140603.jpg" let destination: DownloadRequest.DownloadFileDestination = { let searchPathDirectory: FileManager.SearchPathDirectory = .cachesDirectory let searchPathDomain: FileManager.SearchPathDomainMask = .userDomainMask return DownloadRequest.suggestedDownloadDestination(for: searchPathDirectory, in: searchPathDomain) }() func testThatImmediatelyCancelledDownloadDoesNotHaveResumeDataAvailable() { // Given let expectation = self.expectation(description: "Download should be cancelled") var response: DefaultDownloadResponse? // When let download = Alamofire.download(urlString, to: destination, withMethod: .get) .response { resp in response = resp expectation.fulfill() } download.cancel() waitForExpectations(timeout: timeout, handler: nil) // Then XCTAssertNotNil(response?.request) XCTAssertNil(response?.response) XCTAssertNil(response?.destinationURL) XCTAssertNil(response?.resumeData) XCTAssertNotNil(response?.error) XCTAssertNil(download.resumeData) } func testThatCancelledDownloadResponseDataMatchesResumeData() { // Given let expectation = self.expectation(description: "Download should be cancelled") var response: DefaultDownloadResponse? // When let download = Alamofire.download(urlString, to: destination, withMethod: .get) download.downloadProgress { _, _, _ in download.cancel() } download.response { resp in response = resp expectation.fulfill() } waitForExpectations(timeout: timeout, handler: nil) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNil(response?.destinationURL) XCTAssertNotNil(response?.resumeData) XCTAssertNotNil(response?.error) XCTAssertNotNil(download.resumeData, "resume data should not be nil") if let responseResumeData = response?.resumeData, let resumeData = download.resumeData { XCTAssertEqual(responseResumeData, resumeData) } else { XCTFail("response resume data or resume data was unexpectedly nil") } } func testThatCancelledDownloadResumeDataIsAvailableWithJSONResponseSerializer() { // Given let expectation = self.expectation(description: "Download should be cancelled") var response: DownloadResponse? // When let download = Alamofire.download(urlString, to: destination, withMethod: .get) download.downloadProgress { _, _, _ in download.cancel() } download.responseJSON { resp in response = resp expectation.fulfill() } waitForExpectations(timeout: timeout, handler: nil) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNil(response?.destinationURL) XCTAssertNotNil(response?.resumeData) XCTAssertEqual(response?.result.isFailure, true) XCTAssertNotNil(response?.result.error) XCTAssertNotNil(download.resumeData) } }