3

我有一个函数应该获得两个实际参数进行测试。这两个值都应由 Arbitrary 实例创建,因为它们需要具有某种不能完全任意的良好格式。
所以我创建了以下代码

let updating (x:SomeType) (y:SomeOtherType) =
    let result = update x y
    result.someProp = x.someProp
    && result.otherProp = y.otherProp

let arbSomeType = 
    Arb.generate<SomeType> 
        |> Gen.filter fun x -> x.checkSomeStuff
        |> Arb.fromGen

let arbSomeType = 
    Arb.generate<SomeOtherType> 
        |> Gen.filter fun x -> x.checkPropertiesOfThis
        |> Arb.fromGen

但是我现在如何组合这两个 Arbitrary 实例以使它们与测试方法的签名相匹配?

//let prop = Prop.forAll arbSomeType + arbSomeType updating

Check.QuickThrowOnFailure prop
4

1 回答 1

6

给定两种类型,SomeTypeASomeTypeB

type SomeTypeA = 
    { A : obj }

type SomeTypeB = 
    { B : obj }

您可以创建一个Property,其中输入是这两种类型,如下所示:

let prop =
    gen { let! a = Arb.generate<SomeTypeA>
          let! b = Arb.generate<SomeTypeB>
          return a, b }
    |> Arb.fromGen
    |> Prop.forAll
    <| fun (a, b) ->
        // 'a' is SomeTypeA
        // 'b' is SomeTypeB
        true // Dummy - replace with whatever you want to do with 'a' and 'b'.

您还需要注意,测试方法的签名现在反映了创建的任意 - 成为对上的(非咖喱)函数。

// instead of
let updating (x:SomeType) (y:SomeOtherType) = ...
// do this
let updating (x:SomeType, y:SomeOtherType) = ...

该示例如何工作

  • 计算gen表达式创建类型的生成器Gen<SomeTypeA * SomeTypeB>
  • 从该生成器创建一个Arbitrary<SomeTypeA * SomeTypeB>实例
  • 最后,从任意通过创建(QuickCheck/FsCheck)属性Prop.forAll

它总是相同的路径:

Generator[/optional Shrinker] -> Arbitrary -> Property -> <your_code>

希望有帮助。

于 2016-10-06T12:33:40.703 回答