1

我今天开始使用 BDD 方法编写 iOS 单元测试。我有一个关于guard语句和达到 100% 代码覆盖率的问题。

我有以下代码,它处理对象的Data转换Customer

internal final class func customer(from data: Data) -> Customer? {
    do {
        guard let jsonDictionary = try JSONSerialization.jsonObject(with: data, options: []) as? Dictionary<String, Any> else {
            return nil
        }
        var customerFirstName: String? = nil
        var customerLastName: String
        if let firstName = jsonDictionary["first_name"] as? String {
            customerFirstName = firstName
        }
        guard let lastName = jsonDictionary["last_name"] as? String else {
            return nil
        }
        customerLastName = lastName
        return Customer(firstName: customerFirstName, lastName: customerLastName)
    } catch {
        return nil
    }
}

创建我们的后端时,一些客户只获得了一个姓氏,其中包含他们的名字和姓氏。这就是为什么客户的名字是可选的;他们的全名可能是last_name.

在我的代码中,客户的名字是可选的,而他们的姓氏是必需的。如果从网络请求收到的 JSON 中没有返回他们的姓氏,那么我不会创建客户。此外,如果Data无法序列化为Dictionary,则不会创建客户。

我有两个 JSON 文件,它们都包含我用来测试这两个场景的客户信息。

一个在 JSON 中不包含名字:

{
    "first_name": null,
    "last_name": "Test Name",
}

另一个包含 JSON 中的名字:

{
    "first_name": "Test",
    "last_name": "Name",
}

在我的单元测试中,使用 Quick 和 Nimble,我在Customer名字不可用时处理了 a 的创建:

override func spec() {
    super.spec()
    let bundle = Bundle(for: type(of: self))
    describe("customer") {
        context("whenAllDataAvailable") {
            it("createsSuccessfully") {
                let path = bundle.path(forResource: "CustomerValidFullName", ofType: "json", inDirectory: "ResponseStubs")!
                let url = URL(fileURLWithPath: path)
                let data = try! Data(contentsOf: url)
                let customer = DataTransformer.customer(from: data)
                expect(customer).toNot(beNil())
            }
        }
        context("whenMissingLastName") {
            it("createsUnsuccessfully") {
                let path = bundle.path(forResource: "CustomerMissingLastName", ofType: "json", inDirectory: "ResponseStubs")!
                let url = URL(fileURLWithPath: path)
                let data = try! Data(contentsOf: url)
                let customer = DataTransformer.customer(from: data)
                expect(customer).to(beNil())
            }
        }
    }
}

这可确保我Customer在返回的 JSON 中缺少或存在第一个名称时创建一个。

当我的代码由于数据能够转换为有效的 JSON 对象而没有命中语句的else子句时,如何使用 BDD 达到此方法的 100% 代码覆盖率?guard我是否应该只添加另一个.json包含无法转换为 JSON 对象的数据的文件以确保Customer未创建 a 以及.json包含缺失的文件last_name以确保Customer未创建 a ?

我只是在想“100% 代码覆盖率”的概念吗?我什至需要测试语句的else子句吗?guard我什至有使用 BDD 方法的适当方法吗?

4

2 回答 2

1

只需编写你想要的任何 JSON 格式——你能想到的各种错误格式。例子:

  • 您可以使用不正确的 JSON 来处理您的异常处理。
  • 您可以首先guard使用 JSON 数组而不是字典。

俗话说,你只需要覆盖你想要正确的代码。

TDD 和 BDD 是相关的。在 TDD 中,您将首先编写一个失败的测试。然后,您将编写尽可能快地通过该测试的代码。最后,您将清理代码以使其更好。看起来您是在事后添加测试。

顺便说一句,如果您不使用外部文件,而是将 JSON 直接放入您的测试中,您的测试会更加清晰。这是一个截屏视频,展示了我如何对 JSON 转换的开始进行 TDD。截屏视频使用 Objective-C,但原理相同:https ://qualitycoding.org/tdd-json-parsing/

于 2017-01-02T03:29:52.593 回答
0

if let 100% 的代码覆盖率。

return有时,强制准备格式错误的对象以强制执行命中或return nilguard语句中使用是不可行的。

当您集成了第 3 方 SDK 并且在该方法中创建了相应的第 3 方对象时,就会出现这种情况。例如 :

func aMethod() {
   guard let response = 3rdPartyResponse<3rdPartyInput>.createControl(with: .create(with: .someCase)) as? 3rdPartyResponse<3rdPartyInput> else { return }
}

在这种情况下,它非常困难,有时无法击中回报。

但是如果代码覆盖率是主要标准,您可以在这种情况下使用if let If let 确实提供 100% 的代码覆盖率

于 2021-01-19T22:32:22.680 回答