skip to Main Content

I have a simple rest endpoint built on PHP that works on postman and browsers.

https://someserver.com/api/endpoint.php?name=hello

But the code fails to execute when invoked from Swift.
Upon Investigation, I found out that Postman and Browser sends the Host Header whereas the Swift URLRequest doesn’t. When you don’t send the HOST Header, the php server refuses to accept the incoming request and I believe the request is rejected at the web server level. I tried using the

request.addValue("127.0.0.1", "Host")

but it still didn’t work. Any help is appreciated.

Full Code (Swift 4)

let url = urlInput.addingPercentEncoding(withAllowedCharacters: .urlQueryAllowed)!
        print("URL is: " + url)
        var request = URLRequest(url: URL(string: url)!)
        request.httpMethod = "GET"
            request.addValue(<#T##value: String##String#>, forHTTPHeaderField: <#T##String#>)
        let session = URLSession.shared
        let task = session.dataTask(with: request, completionHandler: { data, response, error -> Void in
            if let httpResponse = response as? HTTPURLResponse {
                
                let respStr = String(data: data!, encoding: String.Encoding.utf8) as String?
                print ("Response is (respStr ?? "")")
                print("statusCode: (httpResponse.statusCode)")
                
                let data: Data? = respStr?.data(using: .utf8)
}
})

2

Answers


  1. Chosen as BEST ANSWER

    I figured out that the issue happens only when you hit a php server. The php web server looks for some header and cant find it when the call is made from Swift. I even copied the working code from Postman and results were the same.

    The solution I went with was to create a proxy server using firebase cloud function. Swift code would hit a firebase cloud function which talks to php and sends the response to swift.


  2. Swift is ignoring the fact that you set the header, as it typically should. See the docs about Swift headers and why it’s ignoring your Host header.

    The URL Loading System handles various aspects of the HTTP protocol 
    for you (HTTP 1.1 persistent connections, proxies, authentication, 
    and so on). As part of this support, the URL Loading System takes 
    responsibility for certain HTTP headers:
    
    Content-Length
    
    Authorization
    
    Connection
    
    Host
    
    Proxy-Authenticate
    
    Proxy-Authorization
    
    WWW-Authenticate
    
    If you set a value for one of these reserved headers, the system may 
    ignore the value you set, or overwrite it with its own value, or 
    simply not send it. Moreover, the exact behavior may change over 
    time. To avoid confusing problems like this, do not set these headers 
    directly.
    

    The documentation is here:

    https://developer.apple.com/documentation/foundation/nsurlrequest#1776617

    I have seen this before with Java – there is a workaround in Java depending on the library you are using, but I don’t know about Swift.

    Login or Signup to reply.
Please signup or login to give your own answer.
Back To Top
Search