使用scalacheck测试有效的状态转换

Pab*_*dez 5 scala state-machine scalacheck property-based-testing

假设我有这个课程:

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)
  }
}
Run Code Online (Sandbox Code Playgroud)

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

Naz*_*iuk 1

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

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


如果其中一个转换集比其他转换集小得多,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)
}
Run Code Online (Sandbox Code Playgroud)