cast server response in AFError

634 Views Asked by At

What I'm trying to achieve is that I have a NetworkManager that handles the request's to the server, and handle the error through AFError. However sometimes when the server response is 4xx, there is a custom message with that response which I want to show that to the user But don't know how to implement it.

This is my NetworkManager

    static let shared:NetworkManager = {
        return NetworkManager()
    }()
    typealias completionHandler = ((Result<Data, AFError>) ->Void)

    func handleAFrequest(request: DataRequest,completion: @escaping completionHandler) {
        
        request.validate(statusCode: 200..<300)
        request.responseJSON { (response) in
            
            switch response.result {
            case .success(_):
                
                if let data = response.data {
                    completion(.success(data))
                }
                
            case .failure(let error):
                print(error.localizedDescription)
                switch error {
                case .invalidURL(let url):
                    print("Invalid URL: \(url) - \(error.localizedDescription)")
                    completion(.failure(.invalidURL(url: URL)))
                
               case .responseValidationFailed(let reason):
                    print("Response validation failed: \(error.localizedDescription); Reason:\(reason)")
                    completion(.failure(.responseValidationFailed(reason: reason)))
                    

I want to be able to cast server response in addition to the error, and show Message of the response to the user. Server Response example when StatusCode is 4xx:

{
   "data":
          "code":401;
          "Message":"Phone Invalid"
}
2

There are 2 best solutions below

2
On

I have parsed api errors in many of my projects. I believe there is a better alternative to handle the showing or errors if any, to the user. Please see my code, in it, if there is a error I show it in a toast. Showing in a toast is the not focal point but you can see how I handle the error case in my code and it has never failed. Please change the params accordingly to your api call

func postMethod(mylist: [String:Any]) {
    print(K.APIUrl)
    print(K.port)
    AF.request("\(K.urlFromUrlField!):\(K.configPort)/snhttp-01?", method: .put, parameters: mylist)
        .authenticate(username: username, password: password)
        .response { response in
            switch response.result {
            case .success:
                print("\nValidation Successful from put method")
                print(response.result)
                print(response.value as Any)
                
                //get xml code and error msg if any
                
                if let response = response.data{
                    
                    let xml = XML.parse(response)
                    print(xml)
                    print("\nThis is the data sent to the server: \(mylist["data"] ?? "No data in data key of the parameter")" )
                    
                    let code = xml.Response.Code.text ?? "No code value in response"
                    let responseMessage = xml.Response.Message.text ?? "No message returned from server"
                    
                    print("\nCode value from server: \(code)")
                    print("\nResponse message from server: \(responseMessage)")
                }
                
                else{
                    print("\nSuccess block: Request Successfully sent, BUT there was nothing from the server to unwrap! / nothing sent back from the server\nThis is the data sent to the server: \(mylist["data"] ?? "No data in data key of the parameter")")
                }
                
                
            case let .failure(error):
                
                if let response = response.data {
                    let xml = XML.parse(response)
                    
                    let code = xml.Response.Code.text ?? "\nNo code value in response"
                    let responseMessage = xml.Response.Message.text ?? "No message returned from server"
                    
                    print("\nCode value from server: \(code)")
                    print("\nResponse message from server: \(responseMessage)")
                    print(error)
                    
                }
                
                else    {
                    print("\nFailure Block: A connection to the server could not be established")
                    
                }
                
            }}
    
    }

This code parses the xml from the api. However you can discard that and just focus on how I handle the response and consequently the error.

0
On

This is the solution that works for me.

All you need to do is create a custom error type:

struct APIError: Error, Decodable {

    let status: Int?
    let message: String?
    let key: String?

}

Then call Alamofire, which will return an AFDataResponse which you can parse:

 func performRequest<T: Decodable>(route: APIRouter, completion: @escaping (APIResponse<T>) -> Void) {
     let decoder = JSONDecoder()
     decoder.keyDecodingStrategy = .convertFromSnakeCase
     AF.request(route)
       .validate()
       .responseDecodable(decoder: decoder, emptyResponseCodes: [200, 201]) { (response: AFDataResponse<T>) in
           self.parseResponse(response: response, completion: completion)
       }
}

Parsing is done like this:

private func parseResponse<T: Decodable>(response: AFDataResponse<T>, completion: @escaping (APIResponse<T>) -> Void) {
        switch response.result {
        case .success(let data):
            completion(APIResponse<T>.success(data))
        case .failure(let error):    
            if let data = response.data,
            // THIS IS WHERE YOU CAST AFError TO YOUR CUSTOM APIError
               let apiError = try? JSONDecoder().decode(APIError.self, from: data) {
                completion(APIResponse.failure(apiError))
            } else {
                completion(APIResponse.failure(error))
            }
        }
    }

Hope this helps!