Как я могу протестировать сетевой запрос с использованием локального файла json? - PullRequest
0 голосов
/ 04 января 2019

Я пытаюсь найти лучший способ для юнит-тестирования сетевого запроса.Сначала я хотел создать локальный файл с ответом JSON для тестирования, но, похоже, это не работает.См. Мой код ниже.

Я хочу проверить, что я могу получить массив, отличный от nil, из обработчика завершения в функции ниже.

class APIClient {
    let downloader = JSONDownloader() // just a class that creates a new data task

    // what I want to test
    func getArticles(from url: URL?, completion: @escaping([Article]?, Error?) -> ()) {
        guard let url = url else { return }

        let request = URLRequest(url: url)
        let task = downloader.createTask(with: request) { json, error in
            DispatchQueue.main.async {

                // parse JSON
                ...

                completion(articles, nil)
            }
        }

        task.resume()
    }
}

Я пытался проверить, как показано ниже, чтобы нетбезрезультатно.

func testArticleResponseIsNotNil() {
    let bundle = Bundle(for: APIClientTests.self)
    guard let path = Bundle.path(forResource: "response-articles", ofType: "json", inDirectory: bundle.bundlePath) else {
        XCTFail("Missing file: response-articles.json")
        return
    }

    let url = URL(fileURLWithPath: path)
    var articles: [Article]?

    let expectation = self.expectation(description: "Articles")
    let client = APIClient()
    client.getArticles(from: url) { response, error in
        articles = response
        expectation.fulfill()
    }

    wait(for: [expectation], timeout: 5)
    XCTAssertNotNil(articles)
}

Есть идеи, как именно я должен проверить эту функцию?

Редактировать: Это класс JSONDownloader.

class JSONDownloader {

    let session: URLSession

    init(configuration: URLSessionConfiguration) {
        self.session = URLSession(configuration: configuration)
    }

    convenience init() {
        self.init(configuration: .default)
    }

    typealias JSON = [String: AnyObject]

    func createTask(with request: URLRequest, completion: @escaping(JSON?, Error?) -> ()) -> URLSessionDataTask {
        let task = session.dataTask(with: request) { data, response, error in
            guard let httpResponse = response as? HTTPURLResponse else { return }

            if httpResponse.statusCode == 200 {
                if let data = data {
                    do {
                        let json = try JSONSerialization.jsonObject(with: data, options: []) as? JSON
                        completion(json, nil)
                    } catch { completion(nil, error) }
                } else { completion(nil, error) }
            } else { completion(nil, error) }
        }

        return task
    }
}
...