如何使用RxSwift和XCTest测试进入Swift主线程的函数?

时间:2017-03-15 15:43:16

标签: swift unit-testing mvvm rx-swift

我在测试视图时遇到了这个问题:

在我的ViewModel中,我调用异步操作,当响应到达时,我使用PublishSubject在我的视图中产生更改。在我的视图中,我调用DispatchQueue.main.async以隐藏或显示按钮。

视图模型

let refreshButtons = PublishSubject<Bool>(true)
refreshButtons.onNext(true)

查看

model.refreshButtons.asObservable()
                .subscribe(onNext: {
                    [unowned self] success in
                    self.updateButtons(success)
                })
                .addDisposableTo(disposable)

private func updateButtons(_ show:Bool) {
   DispatchQueue.main.async{
      button.isHidden = !show
   }
}

现在我不知道如何对refreshButtons.onNext(true)隐藏或显示我的按钮进行单元测试。

我能想到的解决方案是:

  • 覆盖方法并具有异步期望,但为此我需要将方法公开,我不想要的,或
  • 在我的ViewModel中而不是在视图中调度主队列,这对我来说听起来很奇怪,但我可以。

我该如何解决这个问题? 提前谢谢。

2 个答案:

答案 0 :(得分:2)

您可以使用基于单元测试中的谓词的异步期望来等待查看按钮是否不再隐藏。

func testButtonIsHidden() {
  // Setup your objects
  let view = ...
  let viewModel = ...

  // Define an NSPredicate to test your expectation
  let predicate = NSPredicate(block: { input, _ in
    guard let _view = input as? MyView else { return false }
    return _view.button.isHidden == true
  })


  // Create an expectation that will periodically evaluate the predicate 
  // to decided whether it's fulfilled or not
  _ = expectation(for: predicate, evaluatedWith: view, handler: .none)

  // Call the method that should generate the behaviour you are expecting.
  viewModel.methodThatShouldResultInButtonBeingHidden()

  // Wait for the 
  waitForExpectationsWithTimeout(1) { error in
    if let error = error {
      XCTFail("waitForExpectationsWithTimeout errored: \(error)")
    }
  }
}

值得注意的是,您传递给NSPredicate的值应为class。这是因为类是通过引用传递的,因此谓词块内的值将与视图模型触及的值相同。如果您传递的是structenum,它们是通过副本传递的,谓词块将会在运行设置代码时收到该值的副本,它将会总是失败。

如果您更喜欢使用@Randall Wang在其答案中建议的UI测试,那么这篇文章可能对您有用:&#34; How to test UI changes in Xcode 7&#34;。 完全披露,我写了这篇文章。

答案 1 :(得分:0)

首先,您不需要测试私有方法

如果您想测试按钮是否隐藏,请尝试UI测试

这是UI测试的WWDC。

https://developer.apple.com/videos/play/wwdc2015/406/