5

假设我有这个类:

case class Receipt(id: Long, state: String) {
  def transitionTo(newState: String) = {
    if (!canTransitionTo(newState)) {
       throw new IllegalStateExcetion(s"cant transition from $state to $newState")
    }
    this.copy(state = newState)
  }
}

我想canTransitionTo用 scalacheck 命令测试逻辑(为了简单起见,这里不包括在内),但我在如何开始时遇到了一些麻烦。有任何想法吗?

4

1 回答 1

1

一些教程如何使用此框架测试状态机,但它们测试了另一个属性。通常他们Command为每个有效的过渡创建一个并触发scalacheck它们的任何随机组合。此类属性的目标是验证状态机对于任意数量的有效转换是否正常运行。

这种方法不会测试canTransitionTo,因为它假定所有转换都是有效的。测试任何一对状态之间的转换都需要重新实现有效和无效转换的概念scalacheckcanTransitionTo这可能比原始功能更复杂。


如果其中一个过渡集比其他过渡集小得多,scalacheck则可以帮助生成另一个过渡集。例如,如果只有少数有效转换和十分之一无效,那么生成器可以提供帮助。

private val allStates: Gen[String] = Gen.oneOf("State1", "State2", "State3")

private val validTransitions: Set[(String, String)] = Set("State1" -> "State2", "State2" -> "State3", "State3" -> "State1")
private val validTransitionsGen: Gen[(String, String)] = Gen.oneOf(validTransitions.toSeq)

private val invalidTransition: Gen[(String, String)] = for {
  from <- allStates
  to <- allStates
  if !validTransitions.contains(from -> to) //this is reimplementaion of canTransitionTo
} yield from -> to

property("valid transitions") = forAll(validTransitionsGen) { transition =>
  Receipt(0, transition._1).canTransitionTo(transition._2)
}

property("invalid transitions") = forAll(invalidTransition) { transition =>
  !Receipt(0, transition._1).canTransitionTo(transition._2)
}
于 2016-07-31T01:17:46.543 回答