Обработка таймаута с Alamofire

Можно ли добавить обработчик тайм-аута для запроса Alamofire?

В своем проекте я использую Alamofire следующим образом:

init() {
    let configuration = NSURLSessionConfiguration.defaultSessionConfiguration()
    configuration.timeoutIntervalForRequest = 30

    self.alamofireManager = Alamofire.Manager(configuration: configuration)
}

func requestAuthorizationWithEmail(email:NSString, password:NSString, completion: (result: RequestResult) -> Void) {

    self.alamofireManager!.request(.POST, "myURL", parameters:["email": email, "password":password])
        .responseJSON { response in
            switch response.result {
            case .Success(let JSON):
                //do json stuff
            case .Failure(let error):
                print("\n\nAuth request failed with error:\n \(error)")
                completion(result: .ConnectionFailed)
            }
    }
}

РЕДАКТИРОВАТЬ:

запросить сообщение об ошибке

Ошибка домена =NSURLErrorDomain Code=-1001 "Время ожидания истекло". UserInfo={NSUnderlyingError=0x7fc10b937320 {Error Domain=kCFErrorDomainCFNetwork Code=-1001 "(нуль)" UserInfo={_kCFStreamErrorCodeKey=-2102, _kCFStreamErrorDomainKey=4}}, NSErrorFailingURLStringKey= URL, NSErrorFailingURLKey = URL, _kCFStreamErrorDomainKey=4, _kCFStreamErrorCodeKey=-2102, NSLocalizedDescription= Время запроса истекло.}

11 ответов

Решение

Вы можете сравнить error.code и если он равен -1001 который NSURLErrorTimedOut тогда вы знаете, что это был тайм-аут.

Swift 3, Alamofire 4.0.1

let manager = Alamofire.SessionManager.default
manager.session.configuration.timeoutIntervalForRequest = 120

manager.request("yourUrl", method: .post, parameters: ["parameterKey": "value"])
        .responseJSON {
            response in
            switch (response.result) {
            case .success:
                //do json stuff
                break
            case .failure(let error):
                if error._code == NSURLErrorTimedOut {
                    //HANDLE TIMEOUT HERE
                }
                print("\n\nAuth request failed with error:\n \(error)")
                break
            }
        }

Swift 2.2

self.alamofireManager!.request(.POST, "myURL", parameters:["email": email, "password":password])
.responseJSON { response in
    switch response.result {
        case .Success(let JSON):
            //do json stuff
        case .Failure(let error):
            if error._code == NSURLErrorTimedOut {
               //HANDLE TIMEOUT HERE
            }
         print("\n\nAuth request failed with error:\n \(error)")
         completion(result: .ConnectionFailed)
     }
}

Свифт 3

Принятый ответ не работал для меня.

После долгих исследований мне это понравилось.

let manager = Alamofire.SessionManager.default
manager.session.configuration.timeoutIntervalForRequest = 120

manager.request("yourUrl", method: .post, parameters: ["parameterKey": "value"])

Swift 3, Alamofire 4.5.0

Я хотел установить одинаковое время ожидания для каждого HTTP-вызова в моем проекте.

Основная идея заключается в том, чтобы объявить Session Manager Alamofire в качестве глобальной переменной. Затем, чтобы создать переменную URLSessionConfiguration, установите время ожидания в секундах и назначьте его менеджеру.

Каждый вызов в проекте может использовать этот настроенный менеджер сеансов.

В моем случае глобальная переменная Alamofire Session Manager была установлена ​​в файле AppDelegate (глобально), а ее конфигурация управлялась с помощью метода didFinishLaunchingWithOptions

AppDelegate.swift

import UIKit

var AFManager = SessionManager()

@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {

    var window: UIWindow?

    func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey: Any]?) -> Bool {

        let configuration = URLSessionConfiguration.default
        configuration.timeoutIntervalForRequest = 4 // seconds
        configuration.timeoutIntervalForResource = 4 //seconds
        AFManager = Alamofire.SessionManager(configuration: configuration)

        return true
    }
    ...
}

Отныне функция запроса Alamofire может вызываться из любой части приложения с помощью afManager.

Например:

AFManager.request("yourURL", method: .post, parameters: parameters, encoding: JSONEncoding.default).validate().responseJSON { response in
    ...
}

Swift 5, Alamofire 5

Самый чистый способ, который я нашел, который работает с последней версией Alamofire, заключается в следующем:

AF.request(url).response { (dataResponse: AFDataResponse<Data?>) in
    switch dataResponse.result {
    case .success(let data):
        // succes path
    case .failure(let error):
        switch error {
        case .sessionTaskFailed(let urlError as URLError) where urlError.code == .timedOut:
            print("Request timeout!")
        default:
            print("Other error!")
        }
    }
}

Swift 3.x

class NetworkHelper {
    static let shared = NetworkHelper()
    var manager: SessionManager {
        let manager = Alamofire.SessionManager.default
        manager.session.configuration.timeoutIntervalForRequest = 10
        return manager
    }
    func postJSONData( withParams parameters: Dictionary<String, Any>, toUrl urlString: String, completion: @escaping (_ error: Error,_ responseBody: Dictionary<String, AnyObject>?)->()) {
        manager.request(urlString, method: .post, parameters: parameters, encoding: JSONEncoding.default).responseJSON { response in 
            if let error = response.result.error {
                if error._code == NSURLErrorTimedOut {
                    print("Time out occurs!")
                }
            }
        }
    }
}

Swift 3.x

Принятый ответ мне тоже не помог.

Это работа для меня!

let url = URL(string: "yourStringUrl")!
var urlRequest = URLRequest(url: url)
urlRequest.timeoutInterval = 5 // or what you want

И после:

Alamofire.request(urlRequest).response(completionHandler: { (response) in
    /// code here
}

Swift 4

Эта функция "мой путь и время ожидания" работоспособна, в то же время практикуется синглтон для класса API. ссылка отсюда

class Auth {
    static let api = Auth()
    private var alamoFire: SessionManager?

    private init() {}

    func headers() -> HTTPHeaders {
        return [
            "Accept": "XXX",
            "Authorization": "XXX",
            "Content-Type": "XXX"
        ]
    }

    func alamofireConfig(timeout lhs: TimeInterval) {
        let configuration = URLSessionConfiguration.default
        configuration.timeoutIntervalForRequest = lhs
        configuration.timeoutIntervalForResource = lhs
        alamoFire = Alamofire.SessionManager(configuration: configuration, delegate: SessionDelegate(), serverTrustPolicyManager: nil)
    }

    func querySample() {

        alamofireConfig(timeout: 10)
        alamoFire?.request("api_post_url", method: .post, parameters: ["parametersKey": "value"], encoding: JSONEncoding.default, headers: headers())
            .responseJSON(completionHandler: { (response) in
            switch response.result {
            case .success(let value):
                // do your statement
            case .failure(let error):
                if error._code == NSURLErrorTimedOut {
                    // timeout error statement
                } else {
                    // other error statement
                }
            }
        })
    }

    func queryOtherSample() {

        alamofireConfig(timeout: 5)
        alamoFire?.request("api_get_url", method: .get, parameters: nil, encoding: JSONEncoding.default, headers: headers())
            .responseJSON(completionHandler: { (response) in
            switch response.result {
            case .success(let value):
                // do your statement
            case .failure(let error):
                if error._code == NSURLErrorTimedOut {
                    // timeout error statement
                } else {
                    // other error statement
                }
            }
        })
    }

}

Для пользователей Swift 3.x / Swift 4.0 / Swift 5.0 с Alamofire >= 5.0

Используется модификатор запроса для увеличения и уменьшения интервала тайм-аута.

Методы создания запросов Alamofire предлагают наиболее общие параметры для настройки, но иногда их просто недостаточно. URLRequests, созданные на основе переданных значений, можно изменить с помощью закрытия RequestModifier при создании запросов. Например, чтобы установить для параметра timeoutInterval URLRequest значение 120 секунд, измените запрос в закрытии.

       var manager = Session.default
 manager.request(urlString, method: method, parameters: dict, headers: headers, requestModifier: { $0.timeoutInterval = 120 }).validate().responseJSON { response in

ИЛИ ЖЕ

RequestModifiers также работают с синтаксисом завершающего закрытия.

       var manager = Session.default
     manager.request("https://httpbin.org/get") { urlRequest in
    urlRequest.timeoutInterval = 60
    urlRequest.allowsConstrainedNetworkAccess = false
}
.response(...)

Вы также можете проверить это здесь

Сделайте расширение SessionManager и напишите общедоступную статическую переменную, подобную этой, "requestTimeOutInterval", это общедоступная переменная. у него есть время.

extension SessionManager {
    public static let custom: SessionManager = {
        let configuration = URLSessionConfiguration.default
        configuration.timeoutIntervalForRequest = requestTimeOutInterval
        configuration.httpAdditionalHeaders = SessionManager.defaultHTTPHeaders

        return SessionManager(configuration: configuration)
    }()
}

Свифт 5.0, Аламофайр 5.4.2

Код ошибки при истечении времени ожидания всегда равен NSURLErrorTimedOut, поэтому я пытаюсь получить объект Error из AFError и воспроизвести его в NSError.

      extension AFError {
    var isTimeout: Bool {
        if isSessionTaskError,
           let error = underlyingError as NSError?,
           error.code == NSURLErrorTimedOut //-1001
        {
            return true
        }
        return false
    }
}

Вызов при закрытии ответа.

      let request = URLRequest(url: URL(string: "https://httpbin.org/delay/10")!, cachePolicy: .useProtocolCachePolicy, timeoutInterval: 2)
AF.request(request).responseString(completionHandler: { response in
    switch response.result {
    case .success(_):
        print("success")
    case .failure(let error):
        if error.isTimeout {
            print("Timeout!")
        }
    }
})

В Alamofire 5.5 SessionManager был переименован в Session.

Вот ссылка на документацию https://github.com/Alamofire/Alamofire/blob/master/Documentation/Alamofire%205.0%20Migration%20Guide.md#breaking-api-changes

Также пример пользователя

      let manager = Alamofire.Session.default
        manager.session.configuration.timeoutIntervalForRequest = 15
        let headers: HTTPHeaders? = token == nil ? nil : [.authorization(bearerToken: token!),.accept("application/json")]
        manager.request(path, method: method, parameters: parameter, headers: headers).responseJSON { (response) in
            switch response.result {
            case .success:

            case .failure:

            }
        }
Другие вопросы по тегам