In FsCheck, how to generate a test record with non-negative fields?

前端 未结 2 736
予麋鹿
予麋鹿 2020-12-20 22:20

In F#, I have a record with a few fields:

    type myRecord = { a:float; b:float; c:float }

I am using FsCheck to test some properties whic

相关标签:
2条回答
  • 2020-12-20 22:59

    Try this:

    type Generators = 
        static member arbMyRecord =
            fun (a,b,c) -> { myRecord.a = a; b = b; c = c }
            <!> (Arb.generate<float> |> Gen.suchThat ((<) 0.) |> Gen.three)
            |> Arb.fromGen
    
    Arb.register<Generators>() |> ignore
    Check.Quick verify_this_property
    

    The <!> is an infix map, useful for applicative style. This is an equivalent generator:

    type Generators = 
        static member arbMyRecord =
            Arb.generate<float> 
            |> Gen.suchThat ((<) 0.) 
            |> Gen.three
            |> Gen.map (fun (a,b,c) -> { myRecord.a = a; b = b; c = c })
            |> Arb.fromGen
    

    If you don't want to globally register your generator, you can use forAll:

    Check.Quick (forAll Generators.arbMyRecord verify_this_property)
    

    Shrinking left as an exercise ;)

    0 讨论(0)
  • 2020-12-20 23:02

    You can avoid creating custom generator by using FsCheck conditional properties

    let verify_this_property (r:myRecord) =
        (r.a > 0.0 && r.b > 0.0 && r.c > 0.0) ==> lazy (myFunction r = (r.a * r.b) * r.c)
    

    Though this will result in (substantially?) slower execution of the test since FsCheck will have to discard all unsuitable test entries.

    0 讨论(0)
提交回复
热议问题