Вызов веб-службы с помощью httpMethod (POST) в swift 4.1

Позвольте мне сначала описать контроллер представления (дизайн страницы). Я добавил одну кнопку и одну метку. И функция, созданная мной для кнопки. Вот файл viewController.swift:

import UIKit

class ViewController: UIViewController {

@IBOutlet weak var btnHitWebService: UIButton!
@IBOutlet weak var lblResult: UILabel!

override func viewDidLoad() {
    super.viewDidLoad()

}

@IBAction func btnHitWebServiceClick(_ sender: Any) {
    if (getTokenWithPostMethod() == true)
    {
        lblResult.text = "yes, we get token: " + g_token
    }
    else
    {
        lblResult.text = "there is an error, please try again later"
    }
}

func getTokenWithPostMethod() -> Bool{

    var funcResult = false

    var request = URLRequest(url: URL(string: "https://apiABCDEFG.com/token")!)
    request.httpMethod = "POST"
    let postString = "grant_type=password&username=ABCDEF&password=ABCDEF"
    request.httpBody = postString.data(using: .utf8)
    request.addValue("application/x-www-form-urlencoded", forHTTPHeaderField: "Content-Type")
    request.addValue("application/json", forHTTPHeaderField: "Accept")

    let task = URLSession.shared.dataTask(with: request)
    {
        data, response, error in
        guard let data = data, error == nil else
        {
            // check for fundamental networking error
            print("error=\(String(describing: error))")
            return
        }

        if let httpStatus = response as? HTTPURLResponse, httpStatus.statusCode != 200
        {
            // check for http errors
            print("statusCode should be 200, but is \(httpStatus.statusCode)")
            print("response = \(String(describing: response))")
        }

        do
        {
            if let jsonObject = try JSONSerialization.jsonObject(with: data, options: []) as? [String:AnyObject]
            {
                g_token = (jsonObject["access_token"] as? String)!
                print(g_token)
                g_token_type = (jsonObject["token_type"] as? String)!
                print("\n")
                print(g_token_type)
                funcResult = true ;
            }
        }
        catch let error as NSError
        {
            print(error)
        }
    }

    task.resume()
    return funcResult
}

}

Вот моя проблема: когда я нажимаю кнопку, это дает мне токен на экране вывода в xcode (print), но метод getTokenWithPostMethod() превращает меня в ложь. Почему это возвращает меня ложь? В процессе отладки: он переходит от 'let task = ' к 'task.resume()'. Спасибо! (кстати, я не могу поделиться правильной ссылкой, именем пользователя и паролем. Я надеюсь, что эти 3 не нужны, чтобы мы могли решить эту проблему)

2 ответа

Решение

Вы делаете вызов API. Требуется время, чтобы вернуть значение. Здесь обратный оператор не будет ждать ответа. Вы должны использовать либо замыкание, либо делегирование. Вы можете попробовать данный код:

func getTokenWithPostMethod(handler: @escaping (_ funcResult: Bool) -> Void){

    var request = URLRequest(url: URL(string: "https://apiABCDEFG.com/token")!)
    request.httpMethod = "POST"
    let postString = "grant_type=password&username=ABCDEF&password=ABCDEF"
    request.httpBody = postString.data(using: .utf8)
    request.addValue("application/x-www-form-urlencoded", forHTTPHeaderField: "Content-Type")
    request.addValue("application/json", forHTTPHeaderField: "Accept")

    let task = URLSession.shared.dataTask(with: request)
    {
        data, response, error in
        var funcResult = false

        if error == nil, let data = data {
            do
            {
                if let jsonObject = try JSONSerialization.jsonObject(with: data, options: []) as? [String:AnyObject]
                {
                    g_token = (jsonObject["access_token"] as? String)!
                    print(g_token)
                    g_token_type = (jsonObject["token_type"] as? String)!
                    print("\n")
                    print(g_token_type)
                    funcResult = true
                }
            }
            catch let error as NSError
            {
                print(error)
            }
        }
        // edited
        DispatchQueue.main.async {
            handler(funcResult)
        }
    }
    task.resume()
}

Это вернется после получения ответа. Замените вызов этого метода на:

@IBAction func btnHitWebServiceClick(_ sender: Any) {

    getTokenWithPostMethod { (flag) in
        if flag {
            lblResult.text = "yes, we get token: " + g_token
        } else {
            lblResult.text = "there is an error, please try again later"
        }
    }
}

Для более подробной информации, пожалуйста, посетите

Он возвращает false, потому что вы установили var funcResult = false при запуске, поэтому, прежде чем URLSession запросит сервер и вернет только некоторые данные в ответ, ваша функция возвращает значение funcResult.

Вы можете добавить приведенный ниже код в тело задачи URLSession, сразу после попытки catch, вместо добавления его в btnHitWebServiceClick:

if (getTokenWithPostMethod() == true)
    {
        lblResult.text = "yes, we get token: " + g_token
    }
    else
    {
        lblResult.text = "there is an error, please try again later"
    }
Другие вопросы по тегам