单元测试WKNavigationDelegate功能迅速

问题描述

我有一个UIViewController实现了一些WKNavigationDelegate功能,并且我想对这些功能中的逻辑进行单元测试。这是一个示例:

func webView(_ webView: WKWebView,decidePolicyFor navigationAction: WKNavigationAction,decisionHandler: @escaping (WKNavigationActionPolicy) -> Void) {
    guard let url = navigationAction.request.url else {
        decisionHandler(.cancel)
        return
    }

    if url.absoluteString != "https://my-approved-url" {
        decisionHandler(.cancel)
        return
    }
    decisionHandler(.allow)
}

我希望我的单元测试确保基于decisionHandler的{​​{1}},以正确的WKNavigationActionPolicy调用request.url

但是,我不知道如何测试此功能。当我运行测试项目时,在Web视图上调用WKNavigationAction不会触发委托函数。我还尝试过直接调用函数进行测试,但是似乎无法实例化自己的新.load()(.request为只读)。

WKNavigationAction函数中进行单元测试逻辑的正确方法是什么?

解决方法

直接调用委托方法的第二种方法是更好的方法,因为您不依赖于单元测试中WebKit的任何行为。

无法实例化WKNavigationAction的问题可以通过子类化解决。创建一个MockNavigationAction,它返回测试所需的所需请求,如下所示:

final class MockNavigationAction: WKNavigationAction {
    var mockedRequest: URLRequest!
    override var request: URLRequest {
        return mockedRequest
    }

然后在您的单元测试中直接调用委托方法:

func test_AllowsCorrectURL() {
    let sut = ViewController()
    let action = MockNavigationAction()
    action.mockedRequest = URLRequest(url: URL(string: "https://my-approved-url")!)
    let allowExpectation = expectation(description: "Allows action")
    sut.webView(WKWebView(),decidePolicyFor: action) { policy in
        XCTAssertEqual(policy,.allow)
        allowExpectation.fulfill()
    }
    waitForExpectations(timeout: 1.0)
}