如何在Swift 4中测试APIs?

5

我是一个新手,刚开始接触单元测试。我已经测试了很多函数,并且理解了概念,现在我想检查API。这是否可能呢?我猜应该可以。这就是API:

func sendRequest(path: String, params: Dictionary<String, Any>, showSpinner: Bool, completionHandler: @escaping (JSON, Error?) -> Void) {
    if Constants.IS_SIMULATOR {
        print("Path: \(path)")
        print("Params: \(params)")
    }

    if Constants.APP_DEL.reachability?.connection == .none {
        completionHandler(JSON.null, NSError(domain: "No internet", code: 4, userInfo: nil))
        return
    }

    UIApplication.shared.isNetworkActivityIndicatorVisible = true

    if showSpinner {
        HUD.show(.labeledProgress(title: "Loading...", subtitle: "Please wait"))
    }

    if let jsonData = try? JSONSerialization.data(withJSONObject: params, options: .prettyPrinted) {

        let url = NSURL(string: String(format: "%@%@", Constants.TEST_URL, path))!
        let request = NSMutableURLRequest(url: url as URL)
        request.httpMethod = "POST"
        request.httpBody = jsonData
        request.timeoutInterval = 120

        let task = URLSession.shared.dataTask(with: request as URLRequest){ data, response, error in
            DispatchQueue.main.async {
                if error != nil {
                    print(" ........ \(String(describing: error?.localizedDescription))")
                    UIApplication.shared.isNetworkActivityIndicatorVisible = false

                    if showSpinner {
                        HUD.flash(.labeledError(title: "Server issue", subtitle: "Invalid response"), delay: 2.0)
                    }

                    completionHandler(JSON.null, NSError(domain: "Invalid response", code: 420, userInfo: nil))
                    return
                }


                if (data?.isGzipped)! {
                    let decompressedData: Data = try! data!.gunzipped()
                    var json: JSON = JSON.null
                    do {
                        json = try JSON(data: decompressedData)
                    }
                    catch {
                        print(error)
                    }

                    if Constants.IS_SIMULATOR {
                        print("Response: \(json)")
                    }

                    UIApplication.shared.isNetworkActivityIndicatorVisible = false


                    if json["status"].int == 200 {
                        if showSpinner {
                            HUD.flash(.success, delay: 0.5)
                        }
                        completionHandler(json["data"], nil)
                    }
                    else if json["status"].int == 202 {
                        if showSpinner {
                            HUD.hide()
                        }
                        completionHandler(JSON.null, NSError(domain: json["message"].string!, code: json["status"].int!, userInfo: nil))
                    }
                    else if json["status"].int == 310 {
                        if showSpinner {
                            HUD.hide()
                        }
                        completionHandler(json["data"], nil)
                    }
                    else if json["status"].int == 403 {
                        if showSpinner {
                            HUD.hide()
                        }

                        GeneralHelper.sharedInstance.displayAlertMessage(titleStr: "Session expired", messageStr: "Kindly login again.")

                        DispatchQueue.main.asyncAfter(deadline: .now() + 1.0, execute: {

                            let domain = Bundle.main.bundleIdentifier!
                            UserDefaults.standard.removePersistentDomain(forName: domain)
                            UserDefaults.standard.synchronize()
                            Constants.APP_DEL.navC?.popToRootViewController(animated: false)
                        })
                        completionHandler(JSON.null, NSError(domain: json["message"].string!, code: json["status"].int!, userInfo: nil))
                    }
                    else {
                        if showSpinner {
                            HUD.flash(.labeledError(title: "", subtitle: json["message"].string!), delay: 2.0)
                        }
                        completionHandler(JSON.null, NSError(domain: json["message"].string!, code: json["status"].int!, userInfo: nil))
                    }
                }
                else {
                    let backToString = String(data: data!, encoding: String.Encoding.utf8) as String?
                    if Constants.IS_SIMULATOR {
                        print("Invalid response: \(String(describing: backToString))")
                    }
                    UIApplication.shared.isNetworkActivityIndicatorVisible = false


                    if showSpinner {
                        HUD.flash(.labeledError(title: "Server issue", subtitle: "Invalid response"), delay: 2.0)
                    }
                    completionHandler(JSON.null, NSError(domain: "Invalid response", code: 420, userInfo: nil))
                }
            }
        }

        task.resume()
    }
}

因此,为了测试这个,我已经做了以下工作:

func testAPIWorking() {
    params = ["ios_token": "dq6YJkKwEx0:APA91bFeOTfJRFd5G78xMkv3AvjSLA7ey2dJxTZZAtMuuC50CqWILNzNjdgqVpNpDn7R4I0DLoydIVDYKubpGfgfu1bwz1H3VNU4D88ek8PJTAjxrd3CWkW78g0sNv6EZDLlTqUFeNxh", "api_token": "kfRSHL0bVP1fSmxNY3NfEGs8g0ktKCbTsPRRbfarh3a5ISIcZLu3qdK07MJ9H4rJ", "player_id": 8083]

    ServiceHelper.sharedInstance.sendRequest(path: "home", params: self.params, showSpinner: false) { (result, error) in
        if error != nil {
            XCTFail("Fail")
        }
        else {

        }       
    }
}

我在任务处添加了一个断点并打印出了任务,但是当我尝试移动到下一行时,它不会进入Dispatch,而是跳出并停止在task.resume(),因此我无法测试errors或预期的results。有什么帮助吗?


1
天啊,请格式化你的代码。这不是C语言。 - J. Doe
2
你无法进行单元测试。如果你想要测试这个,你需要一个集成测试。你只能对小的、独立的代码单元进行单元测试,而不能对依赖于服务器状态、网络连接等的东西进行测试。将你的实现分解为单独的函数(例如解析),然后你可以对这些部分进行单元测试。 - Sulthan
https://www.raywenderlich.com/709-ios-unit-testing-and-ui-testing-tutorial 希望对您有所帮助。 - Pushp
你是通过Burp还是Charles发送API请求的?两者都有免费版,让你检查应用程序的请求并修改响应,以检查你的错误处理是否良好。 - rustyMagnet
1个回答

5

这里有一个完成处理程序,api调用不是同步的。因此,在测试中,你需要等待结果。在Xcode中,你可以使用XCTestExpectation

例如:

    func testAPIWorking()
        {

            let expectation = XCTestExpectation.init(description: "Your expectation")
            params = ["ios_token": "dq6YJkKwEx0:APA91bFeOTfJRFd5G78xMkv3AvjSLA7ey2dJxTZZAtMuuC50CqWILNzNjdgqVpNpDn7R4I0DLoydIVDYKubpGfgfu1bwz1H3VNU4D88ek8PJTAjxrd3CWkW78g0sNv6EZDLlTqUFeNxh", "api_token": "kfRSHL0bVP1fSmxNY3NfEGs8g0ktKCbTsPRRbfarh3a5ISIcZLu3qdK07MJ9H4rJ", "player_id": 8083]

            ServiceHelper.sharedInstance.sendRequest(path: "home", params: self.params, showSpinner: false) { (result, error) in

                if error != nil
                {
                    XCTFail("Fail")
                }
                // The request is finished, so our expectation
                expectation.fulfill()
            }
            // We ask the unit test to wait our expectation to finish.
            self.waitForExpectations(timeout: 20)
        }

抛出了一个错误 模糊引用成员'expectation(description:)' - Rob13
是的,抱歉我忘记初始化期望值了。请检查新的答案版本。 - Pierre Perrin
是的,我知道我添加了期望,但仍然不知道它是否有效。 - Rob13
你在使用特定的库进行测试吗?你是否已经在Xcode项目中正确设置了你的测试目标(它应该与你的应用程序不同)? - Pierre Perrin
实际上,当我想测试返回 jsonObject 的 API 时,我会测试是否可以正确解析它。例如,在这里,您可以检查您的 Json 的消息。 - Pierre Perrin
显示剩余3条评论

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接