Контекстный тип закрытия '(Data?, URLResponse?, Error?) -> Void' ожидает 3 аргумента, но 1 использовался в теле замыкания

Я следую этому руководству, но получаю эту ошибку:

Контекстный тип закрытия '(Data?, URLResponse?, Error?) -> Void' ожидает 3 аргумента, но 1 использовался в теле замыкания

в строке: urlSession.dataTask(with: url) { (result) in

private func fetchResources<T: Decodable>(url: URL, completion: @escaping (Result<T, APIServiceError>) -> Void) {
    guard var urlComponents = URLComponents(url: url, resolvingAgainstBaseURL: true) else {
        completion(.failure(.invalidEndpoint))
        return
    }
    let queryItems = [URLQueryItem(name: "api_key", value: apiKey)]
    urlComponents.queryItems = queryItems
    guard let url = urlComponents.url else {
        completion(.failure(.invalidEndpoint))
        return
    }

    urlSession.dataTask(with: url) { (result)  in
        switch result {
            case .success(let (response, data)):
                guard let statusCode = (response as? HTTPURLResponse)?.statusCode, 200..<299 ~= statusCode else {
                    completion(.failure(.invalidResponse))
                    return
                }
                do {
                    let values = try self.jsonDecoder.decode(T.self, from: data)
                    completion(.success(values))
                } catch {
                    completion(.failure(.decodeError))
                }
            case .failure(let error):
                completion(.failure(.apiError))
            }
     }.resume()
}

person ioskaveen    schedule 11.01.2021    source источник
comment
@Rob Я исправил параметры, ожидаемые URLSession, но теперь ошибка «Не удается найти «результат» в области видимости» в операторе switch.   -  person ioskaveen    schedule 11.01.2021
comment
Это имеет гораздо больше смысла @Rob большое спасибо!   -  person ioskaveen    schedule 11.01.2021


Ответы (1)


Это закрытие dataTask ожидает три параметра: Data?, URLResponse? и Error?. Ваш код написан так, как если бы это замыкание имело один параметр, Result. Похоже, вы думали, что это Result<(Data, URLResponse), Error>, но это не так. Это перечисление Result используется в нашем собственном коде, но не используется URLSession.

Таким образом, измените dataTask, чтобы использовать замыкание с тремя параметрами:

private func fetchResources<T: Decodable>(url: URL, completion: @escaping (Result<T, APIServiceError>) -> Void) {
    guard var urlComponents = URLComponents(url: url, resolvingAgainstBaseURL: true) else {
        completion(.failure(.invalidEndpoint))
        return
    }
    urlComponents.queryItems = [URLQueryItem(name: "api_key", value: apiKey)]
    guard let url = urlComponents.url else {
        completion(.failure(.invalidEndpoint))
        return
    }

    urlSession.dataTask(with: url) { data, response, error in
        guard error == nil else {
            completion(.failure(.apiError))
            return
        }

        guard 
            let data = data,
            let response = response as? HTTPURLResponse,
            200...299 ~= response.statusCode 
        else {
            completion(.failure(.invalidResponse))
            return
        }

        do {
            let values = try self.jsonDecoder.decode(T.self, from: data)
            completion(.success(values))
        } catch {
            completion(.failure(.decodeError))
        }
    }.resume()
}
person Rob    schedule 11.01.2021