class: center, middle # Much ado about testing Nicolas Rinaudo • [@NicolasRinaudo] --- class: center, middle # Software testing --- ## Testing .center[data:image/s3,"s3://crabby-images/82da1/82da1f2ef5c46f9c1c965fc0da229767a2cc2594" alt="Testing"] --- ## Testing .center[data:image/s3,"s3://crabby-images/4bd19/4bd19525a345ab7519ceb098582d8969974ac9d7" alt="Testing"] --- ## Testing .center[data:image/s3,"s3://crabby-images/53ade/53ade9cfaf6f186f982be8c40557a94e69c0fd75" alt="Testing"] --- ## Testing .center[data:image/s3,"s3://crabby-images/06ffc/06ffcdfcf3fffcb382aff7600fc09828ad17344b" alt="Testing"] --- ## Testing .center[data:image/s3,"s3://crabby-images/53ade/53ade9cfaf6f186f982be8c40557a94e69c0fd75" alt="Testing"] --- ## System Under Test > Write a program that, given a number, prints it. > > But for multiples of three print `Fizz` instead of the number and for the multiples of five print `Buzz`. ```scala def mult3(i: Int) = !(i % 3 > 0) def mult5(i: Int) = !(i % 5 > 0) def fizzBuzz(i: Int) = if mult3(i) then "Fizz" else if mult5(i) then "Buzz" else i.toString ``` --- ## System Under Test > Write a program that, .highlight[given a number], prints it. > > But for multiples of three print `Fizz` instead of the number and for the multiples of five print `Buzz`. ```scala def mult3(i: Int) = !(i % 3 > 0) def mult5(i: Int) = !(i % 5 > 0) def fizzBuzz(`i: Int`) = if mult3(i) then "Fizz" else if mult5(i) then "Buzz" else i.toString ``` --- ## System Under Test > Write a program that, given a number, .highlight[prints it]. > > But for multiples of three print `Fizz` instead of the number and for the multiples of five print `Buzz`. ```scala def mult3(i: Int) = !(i % 3 > 0) def mult5(i: Int) = !(i % 5 > 0) def fizzBuzz(i: Int) = if mult3(i) then "Fizz" else if mult5(i) then "Buzz" else `i.toString` ``` --- ## System Under Test > Write a program that, given a number, prints it. > > But for .highlight[multiples of three] print `Fizz` instead of the number and for the multiples of five print `Buzz`. ```scala def mult3(i: Int) = !(i % 3 > 0) def mult5(i: Int) = !(i % 5 > 0) def fizzBuzz(i: Int) = if `mult3(i)` then "Fizz" else if mult5(i) then "Buzz" else i.toString ``` --- ## System Under Test > Write a program that, given a number, prints it. > > But for multiples of three print .highlight[`Fizz`] instead of the number and for the multiples of five print `Buzz`. ```scala def mult3(i: Int) = !(i % 3 > 0) def mult5(i: Int) = !(i % 5 > 0) def fizzBuzz(i: Int) = if mult3(i) then `"Fizz"` else if mult5(i) then "Buzz" else i.toString ``` --- ## System Under Test > Write a program that, given a number, prints it. > > But for multiples of three print `Fizz` instead of the number and for the .highlight[multiples of five] print `Buzz`. ```scala def mult3(i: Int) = !(i % 3 > 0) def mult5(i: Int) = !(i % 5 > 0) def fizzBuzz(i: Int) = if mult3(i) then "Fizz" else if `mult5(i)` then "Buzz" else i.toString ``` --- ## System Under Test > Write a program that, given a number, prints it. > > But for multiples of three print `Fizz` instead of the number and for the multiples of five print .highlight[`Buzz`]. ```scala def mult3(i: Int) = !(i % 3 > 0) def mult5(i: Int) = !(i % 5 > 0) def fizzBuzz(i: Int) = if mult3(i) then "Fizz" else if mult5(i) then `"Buzz"` else i.toString ``` --- class: center, middle # Example-based testing --- ## Overview .center[data:image/s3,"s3://crabby-images/82da1/82da1f2ef5c46f9c1c965fc0da229767a2cc2594" alt="Testing"] --- ## Overview .center[data:image/s3,"s3://crabby-images/4bd19/4bd19525a345ab7519ceb098582d8969974ac9d7" alt="Testing"] --- ## Overview .center[data:image/s3,"s3://crabby-images/fa510/fa510fc77fd86ae5555848d4787f89c2a96a7e0b" alt="Testing"] --- ## Overview .center[data:image/s3,"s3://crabby-images/10fa4/10fa44d8134390e1ea5a965edd66bcf0b3fc3d3e" alt="Testing"] --- ## Overview .center[data:image/s3,"s3://crabby-images/eb832/eb8326e31b034d085707e4d39ffd3ed157414b2e" alt="Testing"] --- ## Overview .center[data:image/s3,"s3://crabby-images/93631/93631277061f6b3f4dea9212a4894df8c44da60c" alt="Testing"] --- ## Overview .center[data:image/s3,"s3://crabby-images/668c5/668c5832304efa937a3795983cbed19b86da1a97" alt="Testing"] ```scala class ListTests extends AnyFunSuite: test("List(3, 2, 1) should sort to List(1, 2, 3)") { assert(`List(3, 2, 1)`.sorted == List(1, 2, 3)) } ``` --- ## Overview .center[data:image/s3,"s3://crabby-images/3f556/3f5563e50b357f2a382858accff0df49a8817d6a" alt="Testing"] ```scala class ListTests extends AnyFunSuite: test("List(3, 2, 1) should sort to List(1, 2, 3)") { assert(List(3, 2, 1)`.sorted` == List(1, 2, 3)) } ``` --- ## Overview .center[data:image/s3,"s3://crabby-images/684eb/684eb6514a982eaa5e335fe29b99b80c5c1adc64" alt="Testing"] ```scala class ListTests extends AnyFunSuite: test("List(3, 2, 1) should sort to List(1, 2, 3)") { assert(List(3, 2, 1).sorted == `List(1, 2, 3)`) } ``` --- ## FizzBuzz test suite ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(fizzBuzz(3) == "Fizz") } test("n multiple of 5 outputs Buzz") { assert(fizzBuzz(5) == "Buzz") } test("n multiple of neither 3 or 5 outputs n") { assert(fizzBuzz(7) == "7") } ``` --- ## FizzBuzz test suite ```scala class FizzBuzzSuite extends AnyFunSuite: test(`"n multiple of 3 outputs Fizz"`) { assert(fizzBuzz(3) == "Fizz") } test(`"n multiple of 5 outputs Buzz"`) { assert(fizzBuzz(5) == "Buzz") } test(`"n multiple of neither 3 or 5 outputs n"`) { assert(fizzBuzz(7) == "7") } ``` --- ## FizzBuzz test suite ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(fizzBuzz(`3`) == "Fizz") } test("n multiple of 5 outputs Buzz") { assert(fizzBuzz(`5`) == "Buzz") } test("n multiple of neither 3 or 5 outputs n") { assert(fizzBuzz(`7`) == "7") } ``` --- ## FizzBuzz test suite ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(`fizzBuzz`(3) == "Fizz") } test("n multiple of 5 outputs Buzz") { assert(`fizzBuzz`(5) == "Buzz") } test("n multiple of neither 3 or 5 outputs n") { assert(`fizzBuzz`(7) == "7") } ``` --- ## FizzBuzz test suite ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(fizzBuzz(3) == `"Fizz"`) } test("n multiple of 5 outputs Buzz") { assert(fizzBuzz(5) == `"Buzz"`) } test("n multiple of neither 3 or 5 outputs n") { assert(fizzBuzz(7) == `"7"`) } ``` --- ## FizzBuzz test suite ```scala class FizzBuzzSuite extends AnyFunSuite: test(`"n multiple of 3 outputs Fizz"`) { assert(`fizzBuzz(3) == "Fizz"`) } test("n multiple of 5 outputs Buzz") { assert(fizzBuzz(5) == "Buzz") } test("n multiple of neither 3 or 5 outputs n") { assert(fizzBuzz(7) == "7") } ``` --- ## FizzBuzz test suite ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(fizzBuzz(3) == "Fizz") } test(`"n multiple of 5 outputs Buzz"`) { assert(`fizzBuzz(5) == "Buzz"`) } test("n multiple of neither 3 or 5 outputs n") { assert(fizzBuzz(7) == "7") } ``` --- ## FizzBuzz test suite ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(fizzBuzz(3) == "Fizz") } test("n multiple of 5 outputs Buzz") { assert(fizzBuzz(5) == "Buzz") } test(`"n multiple of neither 3 or 5 outputs n"`) { assert(`fizzBuzz(7) == "7"`) } ``` --- ## FizzBuzz test suite ```scala new FizzBuzzSuite().execute() ``` --- ## FizzBuzz test suite ```scala new FizzBuzzSuite().execute() // rs$line$3$FizzBuzzSuite: // - n multiple of 3 outputs Fizz // - n multiple of 5 outputs Buzz // - n multiple of neither 3 or 5 outputs n ``` --- ## Client Demo ```scala fizzBuzz(15) ``` --- ## Client Demo ```scala fizzBuzz(15) // val res1: String = Fizz ``` -- .foreground[data:image/s3,"s3://crabby-images/e16c1/e16c1fa1a1931542cef8a25b2caa5f39c9bed162" alt="Fail"] --- ## Clarifying specifications n % 3 | n % 5 || Output -------|-------||---------- T | T || FizzBuzz T | F || Fizz F | T || Buzz F | F || n --- ## Clarifying specifications .highlight[n % 3] | n % 5 || Output -------|-------||---------- T | T || FizzBuzz T | F || Fizz F | T || Buzz F | F || n --- ## Clarifying specifications n % 3 | .highlight[n % 5] || Output -------|-------||---------- T | T || FizzBuzz T | F || Fizz F | T || Buzz F | F || n --- ## Clarifying specifications n % 3 | n % 5 || .highlight[Output] -------|-------||---------- T | T || FizzBuzz T | F || Fizz F | T || Buzz F | F || n --- ## Clarifying specifications n % 3 | n % 5 || Output -------|-------||---------- .highlight[T] | T || FizzBuzz T | F || Fizz F | T || Buzz F | F || n --- ## Clarifying specifications n % 3 | n % 5 || Output -------|-------||---------- T | .highlight[T] || FizzBuzz T | F || Fizz F | T || Buzz F | F || n --- ## Clarifying specifications n % 3 | n % 5 || Output -------|-------||---------- T | T || .highlight[FizzBuzz] T | F || Fizz F | T || Buzz F | F || n --- ## Fixing FizzBuzz .diff-rm[ ```scala def mult3(i: Int) = !(i % 3 > 0) def mult5(i: Int) = !(i % 5 > 0) def fizzBuzz(i: Int) = * if mult3(i) then `"Fizz"` else if mult5(i) then "Buzz" else i.toString ``` ] --- ## Fixing FizzBuzz .diff-add[ ```scala def mult3(i: Int) = !(i % 3 > 0) def mult5(i: Int) = !(i % 5 > 0) def fizzBuzz(i: Int) = if mult3(i) then * ` if mult5(i) then "FizzBuzz"` * ` else "Fizz"` else if mult5(i) then "Buzz" else i.toString ``` ] --- ## Fixing FizzBuzz ```scala def mult3(i: Int) = !(i % 3 > 0) def mult5(i: Int) = !(i % 5 > 0) def fizzBuzz(i: Int) = if `mult3(i)` then if `mult5(i)` then `"FizzBuzz"` else "Fizz" else if mult5(i) then "Buzz" else i.toString ``` --- ## Fixing FizzBuzz ```scala def mult3(i: Int) = !(i % 3 > 0) def mult5(i: Int) = !(i % 5 > 0) def fizzBuzz(i: Int) = if `mult3(i)` then if mult5(i) then "FizzBuzz" else `"Fizz"` else if mult5(i) then "Buzz" else i.toString ``` --- ## Fixing tests .diff-add[ ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(fizzBuzz(3) == "Fizz") } test("n multiple of 5 outputs Buzz") { assert(fizzBuzz(5) == "Buzz") } test("n multiple of neither 3 or 5 outputs n") { assert(fizzBuzz(7) == "7") } * * `test("n multiple of 3 and 5 outputs FizzBuzz") {` * ` assert(fizzBuzz(15) == "FizzBuzz")` * `}` ``` ] --- ## Fixing tests ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(fizzBuzz(3) == "Fizz") } test("n multiple of 5 outputs Buzz") { assert(fizzBuzz(5) == "Buzz") } test("n multiple of neither 3 or 5 outputs n") { assert(fizzBuzz(7) == "7") } test("n multiple of 3 and 5 outputs FizzBuzz") { assert(fizzBuzz(`15`) == "FizzBuzz") } ``` --- ## Fixing tests ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(fizzBuzz(3) == "Fizz") } test("n multiple of 5 outputs Buzz") { assert(fizzBuzz(5) == "Buzz") } test("n multiple of neither 3 or 5 outputs n") { assert(fizzBuzz(7) == "7") } test("n multiple of 3 and 5 outputs FizzBuzz") { assert(`fizzBuzz`(15) == "FizzBuzz") } ``` --- ## Fixing tests ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(fizzBuzz(3) == "Fizz") } test("n multiple of 5 outputs Buzz") { assert(fizzBuzz(5) == "Buzz") } test("n multiple of neither 3 or 5 outputs n") { assert(fizzBuzz(7) == "7") } test("n multiple of 3 and 5 outputs FizzBuzz") { assert(fizzBuzz(15) == `"FizzBuzz"`) } ``` --- ## Fixing tests ```scala new FizzBuzzSuite().execute() ``` --- ## Fixing tests ```scala new FizzBuzzSuite().execute() // rs$line$7$FizzBuzzSuite: // - n multiple of 3 outputs Fizz // - n multiple of 5 outputs Buzz // - n multiple of neither 3 or 5 outputs n // - n multiple of 3 and 5 outputs FizzBuzz ``` --- ## Client Demo ```scala fizzBuzz(-3) ``` --- ## Client Demo ```scala fizzBuzz(-3) // val res3: String = FizzBuzz ``` -- .foreground[data:image/s3,"s3://crabby-images/e16c1/e16c1fa1a1931542cef8a25b2caa5f39c9bed162" alt="Fail"] --- ## Key takeaways Example-based testing is: -- * very good at the _oracle problem_. -- * very bad at the _test case generation problem_. -- * a little bit discouraging... --- class: center, middle # Generative Testing --- ## Overview .center[data:image/s3,"s3://crabby-images/21ea9/21ea954a07bcfe726249e5d0ea7d5e9e5b663172" alt="Testing"] --- ## Overview .center[data:image/s3,"s3://crabby-images/249fc/249fcbaff18f0c3fc6dbfe27109e96795e1b31e7" alt="Testing"] --- ## Overview .center[data:image/s3,"s3://crabby-images/2cd96/2cd967f9d394bd6ef142462116b8f4e8c5208d3a" alt="Testing"] --- ## Overview .center[data:image/s3,"s3://crabby-images/786fc/786fc8050a80fce890a5aeed5249d0a1d74be92d" alt="Testing"] --- ## Overview .center[data:image/s3,"s3://crabby-images/9366b/9366b7ac22daf225d9ca1090cd9b5554cb51460c" alt="Testing"] --- ## Overview .center[data:image/s3,"s3://crabby-images/a5ee2/a5ee29b5fa743845b479bc8549fab90a6e978c05" alt="Testing"] --- ## Property-Based Testing .center[data:image/s3,"s3://crabby-images/c361d/c361db1a1a23660f6ca382434a4344e08d67bead" alt="Testing"] --- ## Property-Based Testing .center[data:image/s3,"s3://crabby-images/e9fdf/e9fdfe57de217d316448d2b87e3d2850784ff8c8" alt="Testing"] ```scala forAll { `(is: List[Int])` => is.sorted.diff(is).isEmpty } ``` --- ## Property-Based Testing .center[data:image/s3,"s3://crabby-images/86aec/86aec6871ab6ebef915cf56ca0865755c0064b51" alt="Testing"] ```scala forAll { (is: List[Int]) => `is.sorted`.diff(is).isEmpty } ``` --- ## Property-Based Testing .center[data:image/s3,"s3://crabby-images/bace3/bace3d9475d0c821dda6de93a2e202375dde023b" alt="Testing"] ```scala forAll { (is: List[Int]) => is.sorted.`diff(is).isEmpty` } ``` --- ## Property-Based Testing ```scala val propFizzBuzz = forAll { `(i: Int)` => if mult3(i) then if mult5(i) then fizzBuzz(i) == "FizzBuzz" else fizzBuzz(i) == "Fizz" else if mult5(i) then fizzBuzz(i) == "Buzz" else fizzBuzz(i) == i.toString } ``` --- ## Property-Based Testing ```scala val propFizzBuzz = forAll { (i: Int) => if `mult3(i)` then if `mult5(i)` then fizzBuzz(i) == `"FizzBuzz"` else fizzBuzz(i) == "Fizz" else if mult5(i) then fizzBuzz(i) == "Buzz" else fizzBuzz(i) == i.toString } ``` --- ## Property-Based Testing ```scala val propFizzBuzz = forAll { (i: Int) => if `mult3(i)` then if mult5(i) then fizzBuzz(i) == "FizzBuzz" else fizzBuzz(i) == `"Fizz"` else if mult5(i) then fizzBuzz(i) == "Buzz" else fizzBuzz(i) == i.toString } ``` --- ## Property-Based Testing ```scala val propFizzBuzz = forAll { (i: Int) => if mult3(i) then if mult5(i) then fizzBuzz(i) == "FizzBuzz" else fizzBuzz(i) == "Fizz" else if `mult5(i)` then fizzBuzz(i) == `"Buzz"` else fizzBuzz(i) == i.toString } ``` --- ## Property-Based Testing ```scala val propFizzBuzz = forAll { (i: Int) => if mult3(i) then if mult5(i) then fizzBuzz(i) == "FizzBuzz" else fizzBuzz(i) == "Fizz" else if mult5(i) then fizzBuzz(i) == "Buzz" else fizzBuzz(i) == `i.toString` } ``` --- ## Property-Based Testing ```scala propFizzBuzz.check() ``` --- ## Property-Based Testing ```scala propFizzBuzz.check() // + OK, passed 100 tests. ``` --- ## Property-Based Testing ```scala val propFizzBuzz = forAll { (i: Int) => * if mult3(i) then * if mult5(i) then fizzBuzz(i) == "FizzBuzz" * else fizzBuzz(i) == "Fizz" * else if mult5(i) then fizzBuzz(i) == "Buzz" * else fizzBuzz(i) == `i.toString` } ``` --- ## Test oracle .center[data:image/s3,"s3://crabby-images/9e202/9e202b096dc2f73b827e27a24d28372cad179166" alt="Oracle"] --- ## Test oracle .center[data:image/s3,"s3://crabby-images/02b36/02b366190e80f5980430ff08c53668e12863290f" alt="Oracle"] --- ## Test oracle .center[data:image/s3,"s3://crabby-images/745e0/745e0db0d858948c7c0e6b96122d5a4eca6499df" alt="Oracle"] --- ## Test oracle .center[data:image/s3,"s3://crabby-images/6acbc/6acbcd4962cf845b52bc981a1f3f0d01b7fbd404" alt="Oracle"] --- ## Test oracle .center[data:image/s3,"s3://crabby-images/92660/926607dee455d761408516522e27cf737e7ef894" alt="Oracle"] --- ## Test oracle .center[data:image/s3,"s3://crabby-images/f9c90/f9c906f90b46904224d86a42ef1f4e4099b84bc0" alt="Oracle"] --- ## Test oracle .center[data:image/s3,"s3://crabby-images/105c9/105c9defd51da0aef365e01fd71dff8e0d234e0c" alt="Oracle"] ```scala forAll { `(is: List[Int])` => fastSort(is) == is.sorted } ``` --- ## Test oracle .center[data:image/s3,"s3://crabby-images/eab69/eab6986fba8cc1f4edae59e8581e372626fb4259" alt="Oracle"] ```scala forAll { (is: List[Int]) => `fastSort(is)` == is.sorted } ``` --- ## Test oracle .center[data:image/s3,"s3://crabby-images/ebab3/ebab3a02aaf49e8724ab7b5d73256cb365582448" alt="Oracle"] ```scala forAll { (is: List[Int]) => fastSort(is) == `is.sorted` } ``` --- ## Test oracle .center[data:image/s3,"s3://crabby-images/f9c90/f9c906f90b46904224d86a42ef1f4e4099b84bc0" alt="Oracle"] ```scala forAll { (is: List[Int]) => fastSort(is) `==` is.sorted } ``` --- ## Test oracle: use case .center[data:image/s3,"s3://crabby-images/70a55/70a55306cb0e2ec298b6993c36f5f2c4ca79e45d" alt="Oracle"] --- ## Test oracle: use case .center[data:image/s3,"s3://crabby-images/dbd9b/dbd9b491eeeb416244fca6b68a68b12ab7174e61" alt="Oracle"] --- ## Test oracle: use case .center[data:image/s3,"s3://crabby-images/edc6d/edc6da2c8493a2b48e32f45f280219ab031c944c" alt="Oracle"] --- ## Test oracle: use case .center[data:image/s3,"s3://crabby-images/8f2f1/8f2f1a291532d62a18ebf95e7cf6d75ef9afd1e0" alt="Oracle"] --- ## Test oracle: use case .center[data:image/s3,"s3://crabby-images/488cd/488cddf0b608326f6f1e1ece4686f615eb917e94" alt="Oracle"] --- ## Test oracle: use case .center[data:image/s3,"s3://crabby-images/11b44/11b44d0aea4263674453f8970728f4698ed7e37c" alt="Oracle"] --- ## Test oracle: use case .center[data:image/s3,"s3://crabby-images/fe246/fe246851e010af473d03f4d4a32551f3929e91a5" alt="Oracle"] -- .foreground[data:image/s3,"s3://crabby-images/e16c1/e16c1fa1a1931542cef8a25b2caa5f39c9bed162" alt="Fail"] --- ## Test oracle: fizzBuzz .center[data:image/s3,"s3://crabby-images/d6b90/d6b901db925c65d7d41880789d26aeb9feafd9b8" alt="Fail"] --- ## Validity .center[data:image/s3,"s3://crabby-images/6faeb/6faeb23d7abaa12fb2559ccdf6d1d46905d30d60" alt="Validity"] --- ## Validity .center[data:image/s3,"s3://crabby-images/a30b1/a30b12afadbcdaff0a0cd1951f4139a2c415f27c" alt="Validity"] --- ## Validity .center[data:image/s3,"s3://crabby-images/77a7e/77a7e4805f7cfc49e87632a756958b908add21ff" alt="Validity"] --- ## Validity .center[data:image/s3,"s3://crabby-images/0f0d8/0f0d896c77bb368c1979b756a6932a39e3614b68" alt="Validity"] ```scala forAll { `(i: Int)` => math.abs(i) >= 0 } ``` --- ## Validity .center[data:image/s3,"s3://crabby-images/3812b/3812bf8d370823d94063c680f81aca7bfcea6098" alt="Validity"] ```scala forAll { (i: Int) => `math.abs(i)` >= 0 } ``` --- ## Validity .center[data:image/s3,"s3://crabby-images/77a7e/77a7e4805f7cfc49e87632a756958b908add21ff" alt="Validity"] ```scala forAll { (i: Int) => math.abs(i) `>= 0` } ``` --- ## Validity: use case .center[data:image/s3,"s3://crabby-images/eee91/eee9150b805f34989fda49f74728a657d761c81c" alt="Validity"] --- ## Validity: use case .center[data:image/s3,"s3://crabby-images/b593b/b593bf2b660fd2e8f7c081a10bdcfd0342c7a0ea" alt="Validity"] --- ## Validity: use case .center[data:image/s3,"s3://crabby-images/5043f/5043fac460fec3d657196c558380cd450aa3d2e8" alt="Validity"] --- ## Validity: use case .center[data:image/s3,"s3://crabby-images/69f12/69f129923cf9238bcf6172aceed567aba94c02be" alt="Validity"] -- .foreground[data:image/s3,"s3://crabby-images/e16c1/e16c1fa1a1931542cef8a25b2caa5f39c9bed162" alt="Fail"] --- ## Validity: fizzBuzz ```scala val propValidFizzBuzz = forAll { `(i: Int)` => fizzBuzz(i) in Set(i.toString, "Fizz", "Buzz", "FizzBuzz") } ``` --- ## Validity: fizzBuzz ```scala val propValidFizzBuzz = forAll { (i: Int) => `fizzBuzz(i)` in Set(i.toString, "Fizz", "Buzz", "FizzBuzz") } ``` --- ## Validity: fizzBuzz ```scala val propValidFizzBuzz = forAll { (i: Int) => fizzBuzz(i) `in Set(i.toString, "Fizz", "Buzz", "FizzBuzz")` } ``` --- ## Validity: fizzBuzz ```scala propValidFizzBuzz.check() ``` --- ## Validity: fizzBuzz ```scala propValidFizzBuzz.check() // + OK, passed 100 tests. ``` --- ## Involutivity .center[data:image/s3,"s3://crabby-images/496d5/496d5b16494873582d386d7f9e25e21273f59711" alt="Involutivity"] --- ## Involutivity .center[data:image/s3,"s3://crabby-images/b226a/b226a52d3404e41b9a149d6e613387d3c3e66d19" alt="Involutivity"] --- ## Involutivity .center[data:image/s3,"s3://crabby-images/23491/2349182e9c85650ddc93fc2bbded89ae7b93cd38" alt="Involutivity"] --- ## Involutivity .center[data:image/s3,"s3://crabby-images/b01ef/b01ef6dc3ef52138f781d5c3e7253248ec2006fa" alt="Involutivity"] --- ## Involutivity .center[data:image/s3,"s3://crabby-images/06c5b/06c5bd51740bb3f5d920443006275b2b6ebbcaf9" alt="Involutivity"] --- ## Involutivity .center[data:image/s3,"s3://crabby-images/dff7c/dff7c8d38b2f529f56d14e9839dbcd9ad3c6cc81" alt="Involutivity"] --- ## Involutivity .center[data:image/s3,"s3://crabby-images/13603/13603bcfd9c5f53de0afbfcf7cb89c610367c2a4" alt="Involutivity"] --- ## Involutivity .center[data:image/s3,"s3://crabby-images/1860d/1860df381e6a015bbdf18b0679cf029521bfd63b" alt="Involutivity"] ```scala forAll { `(i: Int)` => i.toString.toInt == i } ``` --- ## Involutivity .center[data:image/s3,"s3://crabby-images/03a07/03a0751f0718e3a52737599b9a2d3b94597a2dd7" alt="Involutivity"] ```scala forAll { (i: Int) => i.`toString`.toInt == i } ``` --- ## Involutivity .center[data:image/s3,"s3://crabby-images/a4416/a4416c2f1a0ef8d07c49785eb2acf5cd0c0aa673" alt="Involutivity"] ```scala forAll { (i: Int) => i.toString.`toInt` == i } ``` --- ## Involutivity .center[data:image/s3,"s3://crabby-images/14a1d/14a1d82e6128f38bb4b75cfbfeed1f9b5b658a70" alt="Involutivity"] ```scala forAll { (i: Int) => i.toString.toInt `== i` } ``` --- ## Involutivity: use case ```scala Track( artist = "Iron Maiden", year = 1982, album = "The Number of the Beast", name = "Children of the Damned" ) ``` --- ## Involutivity: use case ```scala Track( artist = "Iron Maiden", year = `1982`, album = "The Number of the Beast", name = "Children of the Damned" ) ``` --- ## Involutivity: use case ```json { "artist" : "Iron Maiden", "year" : `1982`, "album" : "The Number of the Beast", "name" : "Children of the Damned" } ``` --- ## Involutivity: use case ```scala Track( artist = "Iron Maiden", year = `1970`, album = "The Number of the Beast", name = "Children of the Damned" ) ``` -- .foreground[data:image/s3,"s3://crabby-images/e16c1/e16c1fa1a1931542cef8a25b2caa5f39c9bed162" alt="Fail"] --- ## Involutivity: fizzBuzz ```scala val propInvolutiveFizzBuzz = forAll { `(i: Int)` => !(mult3(i) || mult5(i)) ==> (fizzBuzz(i).toInt == i) } ``` --- ## Involutivity: fizzBuzz ```scala val propInvolutiveFizzBuzz = forAll { (i: Int) => `!(mult3(i) || mult5(i))` ==> (fizzBuzz(i).toInt == i) } ``` --- ## Involutivity: fizzBuzz ```scala val propInvolutiveFizzBuzz = forAll { (i: Int) => !(mult3(i) || mult5(i)) `==>` (fizzBuzz(i).toInt == i) } ``` --- ## Involutivity: fizzBuzz ```scala val propInvolutiveFizzBuzz = forAll { (i: Int) => !(mult3(i) || mult5(i)) ==> (`fizzBuzz(i)`.toInt == i) } ``` --- ## Involutivity: fizzBuzz ```scala val propInvolutiveFizzBuzz = forAll { (i: Int) => !(mult3(i) || mult5(i)) ==> (fizzBuzz(i).`toInt` == i) } ``` --- ## Involutivity: fizzBuzz ```scala val propInvolutiveFizzBuzz = forAll { (i: Int) => !(mult3(i) || mult5(i)) ==> (fizzBuzz(i).toInt `== i`) } ``` --- ## Involutivity: fizzBuzz ```scala propInvolutiveFizzBuzz.check() ``` --- ## Involutivity: fizzBuzz ```scala propInvolutiveFizzBuzz.check() // + OK, passed 100 tests. ``` --- ## Idempotence .center[data:image/s3,"s3://crabby-images/d6470/d6470f9096e107a4d28d3e781967ed7abcf0763b" alt="Idempotence"] --- ## Idempotence .center[data:image/s3,"s3://crabby-images/2eb6f/2eb6f206a5c35470b2c7eaa9d5fb9108d36c4a14" alt="Idempotence"] --- ## Idempotence .center[data:image/s3,"s3://crabby-images/4038e/4038e3f3010a7eff47ddb3615a04a1cf9149a760" alt="Idempotence"] --- ## Idempotence .center[data:image/s3,"s3://crabby-images/8a8a4/8a8a4bcbf5c4440873c5ddadc9277ea8459795e3" alt="Idempotence"] --- ## Idempotence .center[data:image/s3,"s3://crabby-images/412eb/412eb2ffd938452e61c93acf5ec91068c6e1f44a" alt="Idempotence"] --- ## Idempotence .center[data:image/s3,"s3://crabby-images/ca621/ca6214517afca2f1e599869da5fedeaa31833fc2" alt="Idempotence"] ```scala forAll { `(is: List[Int])` => is.sorted == is.sorted.sorted } ``` --- ## Idempotence .center[data:image/s3,"s3://crabby-images/557d7/557d762ba5573777ce3711dcceed88e064a5d43e" alt="Idempotence"] ```scala forAll { (is: List[Int]) => `is.sorted` == is.sorted.sorted } ``` --- ## Idempotence .center[data:image/s3,"s3://crabby-images/ffa70/ffa708e30e9fe0b5c53019673f57dc0471799376" alt="Idempotence"] ```scala forAll { (is: List[Int]) => is.sorted == `is.sorted.sorted` } ``` --- ## Idempotence .center[data:image/s3,"s3://crabby-images/412eb/412eb2ffd938452e61c93acf5ec91068c6e1f44a" alt="Idempotence"] ```scala forAll { (is: List[Int]) => is.sorted `==` is.sorted.sorted } ``` --- ## Idempotence: use case `artist` | `year` | `album` | `name` --------------|-------------|-------------------------------------------------- Iron Maiden | 1982 | The Number of the Beast | Children of the Damned Iron Maiden | 1982 | The Number of the Beast | Invaders Iron Maiden | 1982 | The Number of the Beast | Gangland ```scala db.delete( artist = "Iron Maiden", name = "Gangland" ) ``` --- ## Idempotence: use case `artist` | `year` | `album` | `name` --------------|-------------|-------------------------------------------------- Iron Maiden | 1982 | The Number of the Beast | Children of the Damned Iron Maiden | 1982 | The Number of the Beast | Invaders | | | ```scala db.delete( artist = "Iron Maiden", name = "Gangland" ) ``` --- ## Idempotence: use case `artist` | `year` | `album` | `name` --------------|-------------|-------------------------------------------------- Iron Maiden | 1982 | The Number of the Beast | Children of the Damned | | | | | | ```scala db.delete( artist = "Iron Maiden", name = "Gangland" ) ``` -- .foreground[data:image/s3,"s3://crabby-images/e16c1/e16c1fa1a1931542cef8a25b2caa5f39c9bed162" alt="Fail"] --- ## Idempotence: fizzBuzz .center[data:image/s3,"s3://crabby-images/d6b90/d6b901db925c65d7d41880789d26aeb9feafd9b8" alt="Fail"] --- ## Invariance .center[data:image/s3,"s3://crabby-images/6faeb/6faeb23d7abaa12fb2559ccdf6d1d46905d30d60" alt="Invariance"] --- ## Invariance .center[data:image/s3,"s3://crabby-images/04af5/04af5470e4b63b8c4082320735792a87c50761b4" alt="Invariance"] --- ## Invariance .center[data:image/s3,"s3://crabby-images/f9239/f9239d427f5c636699ec3c55f907085b737a514f" alt="Invariance"] --- ## Invariance .center[data:image/s3,"s3://crabby-images/639e6/639e63a3edf9a37391adb593596e362b4655b26c" alt="Invariance"] ```scala forAll { `(is: List[Int])` => is.sorted.diff(is).isEmpty } ``` --- ## Invariance .center[data:image/s3,"s3://crabby-images/f8479/f8479b226d4dcc0c3772391372f51d9645fd6c82" alt="Invariance"] ```scala forAll { (is: List[Int]) => `is.sorted`.diff(is).isEmpty } ``` --- ## Invariance .center[data:image/s3,"s3://crabby-images/f9239/f9239d427f5c636699ec3c55f907085b737a514f" alt="Invariance"] ```scala forAll { (is: List[Int]) => is.sorted.`diff(is).isEmpty` } ``` --- ## Invariance: use case `artist` | `year` | `album` | `name` --------------|-------------|-------------------------------------------------- Iron Maiden | 1982 | The Number of the Beast | Children of the Damned Iron Maiden | 1982 | The Number of the Beast | Invaders --- ## Invariance: use case `artist` | `year` | `album` | `name` --------------|-------------|-------------------------------------------------- İ̵̺͝ṛ̸̣͕̐͜o̴̢̎̿̓̌n̸̥͙̬̣̾́͛̾̌͘ͅ ̶̻͂M̷̗̣̠̠̎̀ȧ̸̛͔͇̐͘į̵̗̞̩̯̫̇̔̾̋̕͝d̵̟͓͆̅̇̋̐̚ͅe̷̯̪̙̼̩̩͐n̴̛̗̟͈͇̖̎̈̕ | 1̶͍͓̣̔̍́̒̒̅͘͜9̷̬̖̹̗̻͍̋͗̈́͠8̶͓̫̹̹̍2̸̨̲̝̏͌ | T̸̜̟͕̩̘̃̇͂͠h̶̦̓͋̇͝e̴̢̝̱̦͎͗͂̅̓͜ͅ ̶̝̬͠͝Ǹ̵̛̘̺̯̃̈́̍ù̷̱̖m̷̗͙̱̥͇̹͑̎̂͊̈́́̌b̸̢̦̪̽̅̂̑e̴̤̫̝̎̈́̇r̵͇̥̃̃̈͋̎̋͗͜ ̴͇̖̼͙̥̲̹̄̀ò̷͉̠̒f̶̲͇̳͛̔̃̽ ̸͍̉̚͝t̶̼̗̬̞̣̂͗͠͠͝h̴̰͒̾̑͐́͝ě̷̠̤͎̳͍̿͆́̕͝ ̸̭̬̊̐̾̍B̷̨̑é̴̡̟͔͖̏̃̔̓͝͠ą̷͉̫͇̫̈́̍̄́̃s̵͍̀̅͂͂͑̌ẗ̷̡̪̙͉͔́͑͗̇̋̑ | C̵͚̣̬̠̞̝̰̐̍̃h̷̥͛͂͋̚̚ï̴̤l̴͉̳̘̝͖̤͋͝d̴̢̲̫̯͙̺̮́̈́̕r̸̢̗̥͇̻͓͗̋̆́ȇ̸̲̉̆n̵̞̙͎̠̲͕̭̅̃̉̂ ̵͔̦͙̯̎͂̑̚͝õ̷̳̱͈̻̤̠̫̽̓̑̊͒f̸̦͋ͅ ̴̹̖̌ṫ̷̻̼͖͗̉̀̑h̴̹͎̰̬̟͊̍ͅe̸̤̱͛́͑ ̸̞̾͊͠D̵̢̜̬̳̍̑͛̄͆̀̅a̴͙̼̤̤͓͇̟̒̈́͋m̷̡̞̬̞̬͆̔̿n̴̨̢̖͉͓̥̂̄͘ë̴̬͚͎̭̭ͅd̸̢͕̼̗̠̂ Ì̷̧̺͓͕̘͓̯̗̬͈̤̗̝̰͜ȓ̵̨̜̟̱̘̟̬̙̯̱͍̩̭̈̆̅̍̌̿́̈́́̚͜͝ͅo̴̡͗͗͛̑̿̾͒͂̎̆n̶͖̳͖͉̗͕̱͔̱̖̫̽̎͜ͅ ̸̢̨̛͇͚͔̘̦̘̱̮̖̿͑̒̉̽̎́̑̚̚̕M̷̜͖̊͗̅̆̏̈́͛̍̂̉́̃̚̕͝ả̷̡̛̫͔̳̞̻͔͉̜̍̅̈́̆͛͒̋̑͠i̵̢̡͎̯̙͇͍̇͊̃̓̚͜ͅd̸̟̤̼̲̈́̎̑͜ͅȩ̸͓̝̘̙͉̼̺̖̳͓̩̽̕ͅn̴͈̈̃̿͝ | 1̴̛͈̣̲̹͗̾͂͆̒̓͛̋͝9̵̨̢̨̻̲̬͕̲͔͖̘̀̿͝ͅ8̷̩̹̣̦̙͖͚̞̮́̉̑̓̉͑̈̾́͊̀̽͛͘̚2̵̡̘̠̼͙̅̇͛̽͒̀̄̾́̋̎͐͛͝ | T̷̢̟̙̥̬̰͎͚͈͕̬͆̋͊͛̂̓̈́͊̌̀͆̅̐͘͝ȟ̴̛̹̝͕͒͗è̵̛̩͇͎̤̪͍͎̳̺͔̖̝͖͊̏̀̽͝ ̵͕͇͌͌́͛̑̏͊̈̔̿̚̕͠͝N̴̤̥̖̠͍̻̤̣͈̩̞̾̍̉͐͋̇̈̈́̓̈́̚ͅu̵̲̹̙̦̞̎̏͗͋̌͐̈́͘͜͝͝m̶̡̙̖̳̖͇̻̰̼̝̖̎̈̈̄́̀̽̈́͘͜͝b̸̡̙͚̙̮̀̀̅́̔́̿̑͆̂͒̕è̷̫͕̗̰̌̈́̏͗͆͂r̶̡̢̨̛̛̙̖̲͔̥̲͕̈́̀͆̀̅̀̔̃̿̉̽͝͝ ̵̨͍͎͈̹̺̟̠̪̼̟̟͔̜̆̈́̈́o̶̡̧̯̠̜̤͊̒̀͆̒̀̋̅̽̕͜͝ͅf̸͓̥̻͚̘̹͈͉̱̆̅̊̿͑͛́͆̽̋͘͜͜ ̶̣̭͙̯͇̦͂̋͆͆̕ţ̴̡̨̭̭̺̣̲̤͎̩̮̮̥̿́̔́̂͂h̵͉̤͛͊͛͐͒͒ė̶̜̰͓̤̼̖̜̣͎̫̑̍̌́̂̈̌͆̍ ̴̘̞̖̳͕̳͑̀̆̑͜B̷̨͈̹̣̘̰͕͉̰̹͗̔ȩ̷̡̠̙̖̱̜̲̤͚̼͓̠͛̾̓͆̅̒͂̅̀̿͘͘ạ̶̢̨̮̩͇͚͓̀̄̐̈́̇̇͊̄̾̕͝͝s̵̡̛͚̭̞͌̋̐̑͑̓̒̋͊͊͌̈̎̚ṱ̴̫̤̩͂͒̅̋̓̓̿̅̈́̎ | I̴̢̦̠͈̊̆͌͝͝ͅn̶̛͈̺̜͇̲̭͓͉̎̽̓̊̈̈́̾̔́̽̇͘v̷͈̘̞̬̼̲̍͊͊̋̓̕ą̷̰͙͙̖̖̤̬͙̃d̷̠̀́̓͆̽̽̽̓͘ȩ̵̤̱͓̫̻̫̯̪̎͌̇̆̄̓ͅr̴̗̹̪̬̘̪̝̰͆̒͆̋̃̃̏̈͑̅͝ş̴͎̭̻͔̘͈͉̟̻͈̯̩͓̔̐ --- ## Invariance: use case `artist` | `year` | `album` | `name` --------------|-------------|-------------------------------------------------- Iron Maiden | 1982 | The Number of the Beast | Children of the Damned Iron Maiden | 1982 | The Number of the Beast | Invaders ```scala forAll { `(cmds: List[Command])` => cmds.forall { cmd => db.execute(cmd) notCorrupt(db) } } ``` --- ## Invariance: use case `artist` | `year` | `album` | `name` --------------|-------------|-------------------------------------------------- Iron Maiden | 1982 | The Number of the Beast | Children of the Damned Iron Maiden | 1982 | The Number of the Beast | Invaders ```scala forAll { (cmds: List[Command]) => cmds.forall { cmd => `db.execute(cmd)` notCorrupt(db) } } ``` --- ## Invariance: use case `artist` | `year` | `album` | `name` --------------|-------------|-------------------------------------------------- Iron Maiden | 1982 | The Number of the Beast | Children of the Damned Iron Maiden | 1982 | The Number of the Beast | Invaders ```scala forAll { (cmds: List[Command]) => cmds.forall { cmd => db.execute(cmd) `notCorrupt(db)` } } ``` --- ## Invariance: use case `artist` | `year` | `album` | `name` --------------|-------------|-------------------------------------------------- İ̵̺͝ṛ̸̣͕̐͜o̴̢̎̿̓̌n̸̥͙̬̣̾́͛̾̌͘ͅ ̶̻͂M̷̗̣̠̠̎̀ȧ̸̛͔͇̐͘į̵̗̞̩̯̫̇̔̾̋̕͝d̵̟͓͆̅̇̋̐̚ͅe̷̯̪̙̼̩̩͐n̴̛̗̟͈͇̖̎̈̕ | 1̶͍͓̣̔̍́̒̒̅͘͜9̷̬̖̹̗̻͍̋͗̈́͠8̶͓̫̹̹̍2̸̨̲̝̏͌ | T̸̜̟͕̩̘̃̇͂͠h̶̦̓͋̇͝e̴̢̝̱̦͎͗͂̅̓͜ͅ ̶̝̬͠͝Ǹ̵̛̘̺̯̃̈́̍ù̷̱̖m̷̗͙̱̥͇̹͑̎̂͊̈́́̌b̸̢̦̪̽̅̂̑e̴̤̫̝̎̈́̇r̵͇̥̃̃̈͋̎̋͗͜ ̴͇̖̼͙̥̲̹̄̀ò̷͉̠̒f̶̲͇̳͛̔̃̽ ̸͍̉̚͝t̶̼̗̬̞̣̂͗͠͠͝h̴̰͒̾̑͐́͝ě̷̠̤͎̳͍̿͆́̕͝ ̸̭̬̊̐̾̍B̷̨̑é̴̡̟͔͖̏̃̔̓͝͠ą̷͉̫͇̫̈́̍̄́̃s̵͍̀̅͂͂͑̌ẗ̷̡̪̙͉͔́͑͗̇̋̑ | C̵͚̣̬̠̞̝̰̐̍̃h̷̥͛͂͋̚̚ï̴̤l̴͉̳̘̝͖̤͋͝d̴̢̲̫̯͙̺̮́̈́̕r̸̢̗̥͇̻͓͗̋̆́ȇ̸̲̉̆n̵̞̙͎̠̲͕̭̅̃̉̂ ̵͔̦͙̯̎͂̑̚͝õ̷̳̱͈̻̤̠̫̽̓̑̊͒f̸̦͋ͅ ̴̹̖̌ṫ̷̻̼͖͗̉̀̑h̴̹͎̰̬̟͊̍ͅe̸̤̱͛́͑ ̸̞̾͊͠D̵̢̜̬̳̍̑͛̄͆̀̅a̴͙̼̤̤͓͇̟̒̈́͋m̷̡̞̬̞̬͆̔̿n̴̨̢̖͉͓̥̂̄͘ë̴̬͚͎̭̭ͅd̸̢͕̼̗̠̂ Ì̷̧̺͓͕̘͓̯̗̬͈̤̗̝̰͜ȓ̵̨̜̟̱̘̟̬̙̯̱͍̩̭̈̆̅̍̌̿́̈́́̚͜͝ͅo̴̡͗͗͛̑̿̾͒͂̎̆n̶͖̳͖͉̗͕̱͔̱̖̫̽̎͜ͅ ̸̢̨̛͇͚͔̘̦̘̱̮̖̿͑̒̉̽̎́̑̚̚̕M̷̜͖̊͗̅̆̏̈́͛̍̂̉́̃̚̕͝ả̷̡̛̫͔̳̞̻͔͉̜̍̅̈́̆͛͒̋̑͠i̵̢̡͎̯̙͇͍̇͊̃̓̚͜ͅd̸̟̤̼̲̈́̎̑͜ͅȩ̸͓̝̘̙͉̼̺̖̳͓̩̽̕ͅn̴͈̈̃̿͝ | 1̴̛͈̣̲̹͗̾͂͆̒̓͛̋͝9̵̨̢̨̻̲̬͕̲͔͖̘̀̿͝ͅ8̷̩̹̣̦̙͖͚̞̮́̉̑̓̉͑̈̾́͊̀̽͛͘̚2̵̡̘̠̼͙̅̇͛̽͒̀̄̾́̋̎͐͛͝ | T̷̢̟̙̥̬̰͎͚͈͕̬͆̋͊͛̂̓̈́͊̌̀͆̅̐͘͝ȟ̴̛̹̝͕͒͗è̵̛̩͇͎̤̪͍͎̳̺͔̖̝͖͊̏̀̽͝ ̵͕͇͌͌́͛̑̏͊̈̔̿̚̕͠͝N̴̤̥̖̠͍̻̤̣͈̩̞̾̍̉͐͋̇̈̈́̓̈́̚ͅu̵̲̹̙̦̞̎̏͗͋̌͐̈́͘͜͝͝m̶̡̙̖̳̖͇̻̰̼̝̖̎̈̈̄́̀̽̈́͘͜͝b̸̡̙͚̙̮̀̀̅́̔́̿̑͆̂͒̕è̷̫͕̗̰̌̈́̏͗͆͂r̶̡̢̨̛̛̙̖̲͔̥̲͕̈́̀͆̀̅̀̔̃̿̉̽͝͝ ̵̨͍͎͈̹̺̟̠̪̼̟̟͔̜̆̈́̈́o̶̡̧̯̠̜̤͊̒̀͆̒̀̋̅̽̕͜͝ͅf̸͓̥̻͚̘̹͈͉̱̆̅̊̿͑͛́͆̽̋͘͜͜ ̶̣̭͙̯͇̦͂̋͆͆̕ţ̴̡̨̭̭̺̣̲̤͎̩̮̮̥̿́̔́̂͂h̵͉̤͛͊͛͐͒͒ė̶̜̰͓̤̼̖̜̣͎̫̑̍̌́̂̈̌͆̍ ̴̘̞̖̳͕̳͑̀̆̑͜B̷̨͈̹̣̘̰͕͉̰̹͗̔ȩ̷̡̠̙̖̱̜̲̤͚̼͓̠͛̾̓͆̅̒͂̅̀̿͘͘ạ̶̢̨̮̩͇͚͓̀̄̐̈́̇̇͊̄̾̕͝͝s̵̡̛͚̭̞͌̋̐̑͑̓̒̋͊͊͌̈̎̚ṱ̴̫̤̩͂͒̅̋̓̓̿̅̈́̎ | I̴̢̦̠͈̊̆͌͝͝ͅn̶̛͈̺̜͇̲̭͓͉̎̽̓̊̈̈́̾̔́̽̇͘v̷͈̘̞̬̼̲̍͊͊̋̓̕ą̷̰͙͙̖̖̤̬͙̃d̷̠̀́̓͆̽̽̽̓͘ȩ̵̤̱͓̫̻̫̯̪̎͌̇̆̄̓ͅr̴̗̹̪̬̘̪̝̰͆̒͆̋̃̃̏̈͑̅͝ş̴͎̭̻͔̘͈͉̟̻͈̯̩͓̔̐ ```scala forAll { (cmds: List[Command]) => cmds.exists { cmd => db.execute(cmd) `notCorrupt(db)` } } ``` -- .foreground[data:image/s3,"s3://crabby-images/e16c1/e16c1fa1a1931542cef8a25b2caa5f39c9bed162" alt="Fail"] --- ## Invariance: fizzBuzz ```scala val propSafeFizzBuzz = forAll { `(i: Int)` => Try(fizzBuzz(i)).isSuccess } ``` --- ## Invariance: fizzBuzz ```scala val propSafeFizzBuzz = forAll { (i: Int) => Try(`fizzBuzz(i)`).isSuccess } ``` --- ## Invariance: fizzBuzz ```scala val propSafeFizzBuzz = forAll { (i: Int) => `Try`(fizzBuzz(i))`.isSuccess` } ``` --- ## Invariance: fizzBuzz ```scala propSafeFizzBuzz.check() ``` --- ## Invariance: fizzBuzz ```scala propSafeFizzBuzz.check() // + OK, passed 100 tests. ``` --- ## Metamorphic relation .center[data:image/s3,"s3://crabby-images/705a1/705a183c5bf1081e2484658faacb248a116999c4" alt="Metamorphic Relation"] --- ## Metamorphic relation .center[data:image/s3,"s3://crabby-images/474c6/474c62ef1e2a0357f54381c63cb7695baba483c4" alt="Metamorphic Relation"] --- ## Metamorphic relation .center[data:image/s3,"s3://crabby-images/f932c/f932c63f831872d6bb82fcf2f82634c935ee685e" alt="Metamorphic Relation"] --- ## Metamorphic relation .center[data:image/s3,"s3://crabby-images/01775/01775341dfbbb339ee7212664eb0c5e89d3b2df3" alt="Metamorphic Relation"] --- ## Metamorphic relation .center[data:image/s3,"s3://crabby-images/27fff/27fff312660cb09bf97fb988eafa4e811cbd2823" alt="Metamorphic Relation"] --- ## Metamorphic relation .center[data:image/s3,"s3://crabby-images/08bba/08bbaeacb5623fab3997643436aac4da914e20a1" alt="Metamorphic Relation"] --- ## Metamorphic relation .center[data:image/s3,"s3://crabby-images/52f18/52f18e7f75b08fbefd5209aed33b02ff51961e99" alt="Metamorphic Relation"] ```scala forAll { `(i: Int)` => mult3(i) == mult3(-i) } ``` --- ## Metamorphic relation .center[data:image/s3,"s3://crabby-images/aa02e/aa02e74dbabb03f034ec3ab529cb447e67ce92d8" alt="Metamorphic Relation"] ```scala forAll { (i: Int) => `mult3(i)` == mult3(-i) } ``` --- ## Metamorphic relation .center[data:image/s3,"s3://crabby-images/648bc/648bc11ac47b85fa685ed11ce0454953b25a6ca0" alt="Metamorphic Relation"] ```scala forAll { (i: Int) => mult3(i) == mult3(`-i`) } ``` --- ## Metamorphic relation .center[data:image/s3,"s3://crabby-images/98376/9837615859c426e3608c02601a7a71ae889bbada" alt="Metamorphic Relation"] ```scala forAll { (i: Int) => mult3(i) == `mult3`(-i) } ``` --- ## Metamorphic relation .center[data:image/s3,"s3://crabby-images/08bba/08bbaeacb5623fab3997643436aac4da914e20a1" alt="Metamorphic Relation"] ```scala forAll { (i: Int) => mult3(i) `==` mult3(-i) } ``` --- ## Metamorphic relation: use case .center[data:image/s3,"s3://crabby-images/0c230/0c2305716b6b752ad3e17b7ce70cf0d153d50ca8" alt="Metamorphic Relation"] --- ## Metamorphic relation: use case .center[data:image/s3,"s3://crabby-images/0c1e2/0c1e2caa8fbf6b4fd093a2f575d1278afce9df90" alt="Metamorphic Relation"] --- ## Metamorphic relation: use case .center[data:image/s3,"s3://crabby-images/e42b5/e42b5b352a9aa644c09e8d6d93a580551ecfdd9d" alt="Metamorphic Relation"] --- ## Metamorphic relation: use case .center[data:image/s3,"s3://crabby-images/21a06/21a06b97c237e5a798d5c87fe7bf33b85e165f9b" alt="Metamorphic Relation"] --- ## Metamorphic relation: use case .center[data:image/s3,"s3://crabby-images/76d06/76d06b55c55a8195eedd07f1c6651e68d4f27ed4" alt="Metamorphic Relation"] --- ## Metamorphic relation: use case .center[data:image/s3,"s3://crabby-images/19fd2/19fd2a8b190cea8327f71c72760d413c346c933d" alt="Metamorphic Relation"] -- .foreground[data:image/s3,"s3://crabby-images/e16c1/e16c1fa1a1931542cef8a25b2caa5f39c9bed162" alt="Fail"] --- ## Metamorphic relation: use case _[Metamorphic testing of RESTful web apis](http://www.lsi.us.es/~segura/files/papers/segura17-tse.pdf)_: * 3 new unique Spotify bugs. * 8 new unique YouTube bugs. --- ## Metamorphic relation: fizzBuzz ```scala val propStableFizz = forAll { `(i: Int)` => fizzBuzz(i).contains("Fizz") ==> fizzBuzz(-i).contains("Fizz") } ``` --- ## Metamorphic relation: fizzBuzz ```scala val propStableFizz = forAll { (i: Int) => `fizzBuzz(i)`.contains("Fizz") ==> fizzBuzz(-i).contains("Fizz") } ``` --- ## Metamorphic relation: fizzBuzz ```scala val propStableFizz = forAll { (i: Int) => fizzBuzz(i).contains("Fizz") ==> fizzBuzz(`-i`).contains("Fizz") } ``` --- ## Metamorphic relation: fizzBuzz ```scala val propStableFizz = forAll { (i: Int) => fizzBuzz(i).contains("Fizz") ==> `fizzBuzz(-i)`.contains("Fizz") } ``` --- ## Metamorphic relation: fizzBuzz ```scala val propStableFizz = forAll { (i: Int) => fizzBuzz(i)`.contains("Fizz")` ==> fizzBuzz(-i).contains("Fizz") } ``` --- ## Metamorphic relation: fizzBuzz ```scala val propStableFizz = forAll { (i: Int) => fizzBuzz(i).contains("Fizz") `==>` fizzBuzz(-i).contains("Fizz") } ``` --- ## Metamorphic relation: fizzBuzz ```scala val propStableFizz = forAll { (i: Int) => fizzBuzz(i).contains("Fizz") ==> fizzBuzz(-i)`.contains("Fizz")` } ``` --- ## Metamorphic relation: fizzBuzz ```scala propStableFizz.check() ``` --- ## Metamorphic relation: fizzBuzz ```scala propStableFizz.check() // ! Falsified after 1 passed tests. // > ARG_0: -1 // > ARG_0_ORIGINAL: -67 ``` --- ## Fixing `fizzBuzz` .diff-rm[ ```scala *def mult3(i: Int) = `!(i % 3 > 0)` *def mult5(i: Int) = `!(i % 5 > 0)` def fizzBuzz(i: Int) = if mult3(i) then if mult5(i) then "FizzBuzz" else "Fizz" else if mult5(i) then "Buzz" else i.toString ``` ] --- ## Fixing `fizzBuzz` .diff-add[ ```scala *def mult3(i: Int) = `i % 3 == 0` *def mult5(i: Int) = `i % 5 == 0` def fizzBuzz(i: Int) = if mult3(i) then if mult5(i) then "FizzBuzz" else "Fizz" else if mult5(i) then "Buzz" else i.toString ``` ] --- ## Fixing `fizzBuzz` ```scala propStableFizz.check() ``` --- ## Fixing `fizzBuzz` ```scala propStableFizz.check() // + OK, passed 100 tests. ``` --- ## Challenging properties ```scala class FizzBuzzProps extends Properties("FizzBuzz"): property("validity") = forAll { (i: Int) => fizzBuzz(i) in Set(i.toString, "Fizz", "Buzz", "FizzBuzz") } property("involutivity") = forAll { (i: Int) => !(mult3(i) || mult5(i)) ==> (fizzBuzz(i).toInt == i) } property("metamorphic") = forAll { (i: Int) => fizzBuzz(i).contains("Fizz") ==> fizzBuzz(-i).contains("Fizz") } property("invariant") = forAll { (i: Int) => Try(fizzBuzz(i)).isSuccess } ``` --- ## Challenging properties ```scala class FizzBuzzProps extends Properties("FizzBuzz"): * property("validity") = forAll { (i: Int) => * fizzBuzz(i) in Set(i.toString, "Fizz", "Buzz", "FizzBuzz") * } property("involutivity") = forAll { (i: Int) => !(mult3(i) || mult5(i)) ==> (fizzBuzz(i).toInt == i) } property("metamorphic") = forAll { (i: Int) => fizzBuzz(i).contains("Fizz") ==> fizzBuzz(-i).contains("Fizz") } property("invariant") = forAll { (i: Int) => Try(fizzBuzz(i)).isSuccess } ``` --- ## Challenging properties ```scala class FizzBuzzProps extends Properties("FizzBuzz"): property("validity") = forAll { (i: Int) => fizzBuzz(i) in Set(i.toString, "Fizz", "Buzz", "FizzBuzz") } * property("involutivity") = forAll { (i: Int) => * !(mult3(i) || mult5(i)) ==> (fizzBuzz(i).toInt == i) * } property("metamorphic") = forAll { (i: Int) => fizzBuzz(i).contains("Fizz") ==> fizzBuzz(-i).contains("Fizz") } property("invariant") = forAll { (i: Int) => Try(fizzBuzz(i)).isSuccess } ``` --- ## Challenging properties ```scala class FizzBuzzProps extends Properties("FizzBuzz"): property("validity") = forAll { (i: Int) => fizzBuzz(i) in Set(i.toString, "Fizz", "Buzz", "FizzBuzz") } property("involutivity") = forAll { (i: Int) => !(mult3(i) || mult5(i)) ==> (fizzBuzz(i).toInt == i) } * property("metamorphic") = forAll { (i: Int) => * fizzBuzz(i).contains("Fizz") ==> fizzBuzz(-i).contains("Fizz") * } property("invariant") = forAll { (i: Int) => Try(fizzBuzz(i)).isSuccess } ``` --- ## Challenging properties ```scala class FizzBuzzProps extends Properties("FizzBuzz"): property("validity") = forAll { (i: Int) => fizzBuzz(i) in Set(i.toString, "Fizz", "Buzz", "FizzBuzz") } property("involutivity") = forAll { (i: Int) => !(mult3(i) || mult5(i)) ==> (fizzBuzz(i).toInt == i) } property("metamorphic") = forAll { (i: Int) => fizzBuzz(i).contains("Fizz") ==> fizzBuzz(-i).contains("Fizz") } * property("invariant") = forAll { (i: Int) => * Try(fizzBuzz(i)).isSuccess * } ``` --- ## Challenging properties ```scala new FizzBuzzProps().check() ``` --- ## Challenging properties ```scala new FizzBuzzProps().check() // + FizzBuzz.validity: OK, passed 100 tests. // + FizzBuzz.involutivity: OK, passed 100 tests. // + FizzBuzz.metamorphic: OK, passed 100 tests. // + FizzBuzz.invariant: OK, passed 100 tests. ``` --- ## Challenging properties .diff-rm[ ```scala def fizzBuzz(i: Int) = if mult3(i) then if mult5(i) then "FizzBuzz" * else `"Fizz"` * else if mult5(i) then `"Buzz"` else i.toString ``` ] --- ## Challenging properties .diff-add[ ```scala def fizzBuzz(i: Int) = if mult3(i) then if mult5(i) then "FizzBuzz" * else `"Buzz"` * else if mult5(i) then `"Fizz"` else i.toString ``` ] --- ## Challenging properties ```scala new FizzBuzzProps().check() ``` --- ## Challenging properties ```scala new FizzBuzzProps().check() // + FizzBuzz.validity: OK, passed 100 tests. // + FizzBuzz.involutivity: OK, passed 100 tests. // + FizzBuzz.metamorphic: OK, passed 100 tests. // + FizzBuzz.invariant: OK, passed 100 tests. ``` -- .foreground[data:image/s3,"s3://crabby-images/e16c1/e16c1fa1a1931542cef8a25b2caa5f39c9bed162" alt="Fail"] --- ## Challenging properties ```scala val propFizz = forAll { (i: Int) => fizzBuzz(i * 3).contains("Fizz") } ``` --- ## Challenging properties ```scala val propFizz = forAll { `(i: Int)` => fizzBuzz(i * 3).contains("Fizz") } ``` --- ## Challenging properties ```scala val propFizz = forAll { (i: Int) => fizzBuzz(`i * 3`).contains("Fizz") } ``` --- ## Challenging properties ```scala val propFizz = forAll { (i: Int) => `fizzBuzz`(i * 3).contains("Fizz") } ``` --- ## Challenging properties ```scala val propFizz = forAll { (i: Int) => fizzBuzz(i * 3).`contains("Fizz")` } ``` --- ## Challenging properties ```scala propFizz.check() ``` --- ## Challenging properties ```scala propFizz.check() // ! Falsified after 0 passed tests. // > ARG_0: 1 // > ARG_0_ORIGINAL: 34 ``` --- ## Key takeaways Property-based testing is: -- * very good at the _test case generation problem_. -- * good at the _oracle problem_. -- * hard to get the hang of. --- class: center, middle # Generative Example-Based Tests --- ## Example-Based Tests ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(fizzBuzz(3) == "Fizz") } test("n multiple of 5 outputs Buzz") { assert(fizzBuzz(5) == "Buzz") } test("n multiple of neither 3 or 5 outputs n") { assert(fizzBuzz(7) == "7") } test("n multiple of 3 and 5 outputs FizzBuzz") { assert(fizzBuzz(15) == "FizzBuzz") } ``` --- ## Example-Based Tests ```scala class FizzBuzzSuite extends AnyFunSuite: * test("n multiple of 3 outputs Fizz") { * assert(fizzBuzz(3) == "Fizz") * } test("n multiple of 5 outputs Buzz") { assert(fizzBuzz(5) == "Buzz") } test("n multiple of neither 3 or 5 outputs n") { assert(fizzBuzz(7) == "7") } test("n multiple of 3 and 5 outputs FizzBuzz") { assert(fizzBuzz(15) == "FizzBuzz") } ``` --- ## Example-Based Tests ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(`fizzBuzz(3) == "Fizz"`) } test("n multiple of 5 outputs Buzz") { assert(fizzBuzz(5) == "Buzz") } test("n multiple of neither 3 or 5 outputs n") { assert(fizzBuzz(7) == "7") } test("n multiple of 3 and 5 outputs FizzBuzz") { assert(fizzBuzz(15) == "FizzBuzz") } ``` --- ## Naive property ```scala val propFizzBuzz3 = fizzBuzz(3) == "Fizz" ``` --- ## Naive property ```scala val propFizzBuzz3 = fizzBuzz(`3`) == "Fizz" ``` --- ## Naive property .diff-add[ ```scala val propFizzBuzz3 = * `forAll(const(3)) { i =>` * ` `fizzBuzz(3) == "Fizz" * `}` ``` ] --- ## Naive property ```scala val propFizzBuzz3 = forAll(const(3)) { `i` => fizzBuzz(3) == "Fizz" } ``` --- ## Naive property ```scala val propFizzBuzz3 = forAll(`const(3)`) { i => fizzBuzz(3) == "Fizz" } ``` --- ## Naive property .diff-rm[ ```scala val propFizzBuzz3 = forAll(const(3)) { i => * fizzBuzz(`3`) == "Fizz" } ``` ] --- ## Naive property .diff-add[ ```scala val propFizzBuzz3 = forAll(const(3)) { i => * fizzBuzz(`i`) == "Fizz" } ``` ] --- ## Naive property ```scala val propFizzBuzz3 = forAll(`const(3)`) { i => fizzBuzz(i) == "Fizz" } ``` --- ## Multiples of 3 ```scala val genMult3: Gen[Int] = `arbitrary[Int]` .map(_ * 3) .filter(_ % 5 != 0) ``` --- ## Multiples of 3 ```scala val genMult3: Gen[Int] = arbitrary[Int] .`map(_ * 3)` .filter(_ % 5 != 0) ``` --- ## Multiples of 3 ```scala val genMult3: Gen[Int] = arbitrary[Int] .map(_ * 3) .`filter(_ % 5 != 0)` ``` --- ## Multiples of 3 .diff-rm[ ```scala val propFizzBuzz3 = * forAll(`const(3)`) { i => fizzBuzz(i) == "Fizz" } ``` ] --- ## Multiples of 3 .diff-add[ ```scala val propFizzBuzz3 = * forAll(`genMult3`) { i => fizzBuzz(i) == "Fizz" } ``` ] --- ## Generic property ```scala val propFizzBuzz3 = forAll(`genMult3`) { i => fizzBuzz(i) == "Fizz" } ``` --- ## Generic property ```scala val propFizzBuzz3 = forAll(genMult3) { i => fizzBuzz(i) == `"Fizz"` } ``` --- ## Generic property .diff-rm[ ```scala *`val propFizzBuzz3` = forAll(genMult3) { i => fizzBuzz(i) == "Fizz" } ``` ] --- ## Generic property .diff-add[ ```scala *`def propFizzBuzz(gen: Gen[Int])` = forAll(genMult3) { i => fizzBuzz(i) == "Fizz" } ``` ] --- ## Generic property .diff-rm[ ```scala def propFizzBuzz(gen: Gen[Int]) = * forAll(`genMult3`) { i => fizzBuzz(i) == "Fizz" } ``` ] --- ## Generic property .diff-add[ ```scala def propFizzBuzz(gen: Gen[Int]) = * forAll(`gen`) { i => fizzBuzz(i) == "Fizz" } ``` ] --- ## Generic property .diff-add[ ```scala *def propFizzBuzz(gen: Gen[Int]`, expected: String`) = forAll(gen) { i => fizzBuzz(i) == "Fizz" } ``` ] --- ## Generic property .diff-rm[ ```scala def propFizzBuzz(gen: Gen[Int], expected: String) = forAll(gen) { i => * fizzBuzz(i) == `"Fizz"` } ``` ] --- ## Generic property .diff-add[ ```scala def propFizzBuzz(gen: Gen[Int], expected: String) = forAll(gen) { i => * fizzBuzz(i) == `expected` } ``` ] --- ## Generic property .diff-rm[ ```scala val propFizzBuzz3 = *` forAll(genMult3) { i =>` *` fizzBuzz(i) == "Fizz"` *` }` ``` ] --- ## Generic property .diff-add[ ```scala val propFizzBuzz3 = *` propFizzBuzz(` *` gen = genMult3,` *` expected = "Fizz"` *` )` ``` ] --- ## Generic property ```scala val propFizzBuzz3 = propFizzBuzz( `gen = genMult3`, expected = "Fizz" ) ``` --- ## Generic property ```scala val propFizzBuzz3 = propFizzBuzz( gen = genMult3, `expected = "Fizz"` ) ``` --- ## Multiples of 5 ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(fizzBuzz(3) == "Fizz") } * test("n multiple of 5 outputs Buzz") { * assert(fizzBuzz(5) == "Buzz") * } test("n multiple of neither 3 or 5 outputs n") { assert(fizzBuzz(7) == "7") } test("n multiple of 3 and 5 outputs FizzBuzz") { assert(fizzBuzz(15) == "FizzBuzz") } ``` --- ## Multiples of 5 ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(fizzBuzz(3) == "Fizz") } test("n multiple of 5 outputs Buzz") { assert(fizzBuzz(`5`) == "Buzz") } test("n multiple of neither 3 or 5 outputs n") { assert(fizzBuzz(7) == "7") } test("n multiple of 3 and 5 outputs FizzBuzz") { assert(fizzBuzz(15) == "FizzBuzz") } ``` --- ## Multiples of 5 ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(fizzBuzz(3) == "Fizz") } test("n multiple of 5 outputs Buzz") { assert(fizzBuzz(5) == `"Buzz"`) } test("n multiple of neither 3 or 5 outputs n") { assert(fizzBuzz(7) == "7") } test("n multiple of 3 and 5 outputs FizzBuzz") { assert(fizzBuzz(15) == "FizzBuzz") } ``` --- ## Multiples of 5 ```scala val genMult5: Gen[Int] = `arbitrary[Int]` .map(_ * 5) .filter(_ % 3 != 0) ``` --- ## Multiples of 5 ```scala val genMult5: Gen[Int] = arbitrary[Int] .`map(_ * 5)` .filter(_ % 3 != 0) ``` --- ## Multiples of 5 ```scala val genMult5: Gen[Int] = arbitrary[Int] .map(_ * 5) .`filter(_ % 3 != 0)` ``` --- ## Multiples of 5 ```scala val propFizzBuzz5 = propFizzBuzz( `gen = genMult5`, expected = "Buzz" ) ``` --- ## Multiples of 5 ```scala val propFizzBuzz5 = propFizzBuzz( gen = genMult5, `expected = "Buzz"` ) ``` --- ## Multiples of neither 3 nor 5 ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(fizzBuzz(3) == "Fizz") } test("n multiple of 5 outputs Buzz") { assert(fizzBuzz(5) == "Buzz") } * test("n multiple of neither 3 or 5 outputs n") { * assert(fizzBuzz(7) == "7") * } test("n multiple of 3 and 5 outputs FizzBuzz") { assert(fizzBuzz(15) == "FizzBuzz") } ``` --- ## Multiples of neither 3 nor 5 ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(fizzBuzz(3) == "Fizz") } test("n multiple of 5 outputs Buzz") { assert(fizzBuzz(5) == "Buzz") } test("n multiple of neither 3 or 5 outputs n") { assert(fizzBuzz(`7`) == "7") } test("n multiple of 3 and 5 outputs FizzBuzz") { assert(fizzBuzz(15) == "FizzBuzz") } ``` --- ## Multiples of neither 3 nor 5 ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(fizzBuzz(3) == "Fizz") } test("n multiple of 5 outputs Buzz") { assert(fizzBuzz(5) == "Buzz") } test("n multiple of neither 3 or 5 outputs n") { assert(fizzBuzz(7) == `"7"`) } test("n multiple of 3 and 5 outputs FizzBuzz") { assert(fizzBuzz(15) == "FizzBuzz") } ``` --- ## Multiples of neither 3 nor 5 ```scala val genMultNone: Gen[Int] = `arbitrary[Int]` .filter(_ % 5 != 0) .filter(_ % 3 != 0) ``` --- ## Multiples of neither 3 nor 5 ```scala val genMultNone: Gen[Int] = arbitrary[Int] .`filter(_ % 5 != 0)` .filter(_ % 3 != 0) ``` --- ## Multiples of neither 3 nor 5 ```scala val genMultNone: Gen[Int] = arbitrary[Int] .filter(_ % 5 != 0) .`filter(_ % 3 != 0)` ``` --- ## Multiples of neither 3 nor 5 ```scala val propFizzBuzzNone = propFizzBuzz( `gen = genMultNone`, expected = ??? ) ``` --- ## Multiples of neither 3 nor 5 ```scala val propFizzBuzzNone = propFizzBuzz( gen = genMultNone, `expected = ???` ) ``` --- ## Multiples of neither 3 nor 5 .diff-rm[ ```scala *def propFizzBuzz(gen: Gen[Int], `expected: String`) = forAll(gen) { i => * fizzBuzz(i) == `expected` } ``` ] --- ## Multiples of neither 3 nor 5 .diff-add[ ```scala *def propFizzBuzz(gen: Gen[Int], `oracle: Int => String`) = forAll(gen) { i => * fizzBuzz(i) == `oracle(i)` } ``` ] --- ## Multiples of neither 3 nor 5 .diff-rm[ ```scala val propFizzBuzzNone = propFizzBuzz( gen = genMultNone, * `expected = ???` ) ``` ] --- ## Multiples of neither 3 nor 5 .diff-add[ ```scala val propFizzBuzzNone = propFizzBuzz( gen = genMultNone, * `oracle = _.toString` ) ``` ] --- ## Multiples of 3 and 5 ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(fizzBuzz(3) == "Fizz") } test("n multiple of 5 outputs Buzz") { assert(fizzBuzz(5) == "Buzz") } test("n multiple of neither 3 or 5 outputs n") { assert(fizzBuzz(7) == "7") } * test("n multiple of 3 and 5 outputs FizzBuzz") { * assert(fizzBuzz(15) == "FizzBuzz") * } ``` --- ## Multiples of 3 and 5 ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(fizzBuzz(3) == "Fizz") } test("n multiple of 5 outputs Buzz") { assert(fizzBuzz(5) == "Buzz") } test("n multiple of neither 3 or 5 outputs n") { assert(fizzBuzz(7) == "7") } test("n multiple of 3 and 5 outputs FizzBuzz") { assert(fizzBuzz(`15`) == "FizzBuzz") } ``` --- ## Multiples of 3 and 5 ```scala class FizzBuzzSuite extends AnyFunSuite: test("n multiple of 3 outputs Fizz") { assert(fizzBuzz(3) == "Fizz") } test("n multiple of 5 outputs Buzz") { assert(fizzBuzz(5) == "Buzz") } test("n multiple of neither 3 or 5 outputs n") { assert(fizzBuzz(7) == "7") } test("n multiple of 3 and 5 outputs FizzBuzz") { assert(fizzBuzz(15) == `"FizzBuzz"`) } ``` --- ## Multiples of 3 and 5 ```scala val genMult15: Gen[Int] = `arbitrary[Int]`. map(_ * 3 * 5) ``` --- ## Multiples of 3 and 5 ```scala val genMult15: Gen[Int] = arbitrary[Int]. `map(_ * 3 * 5)` ``` --- ## Multiples of 3 and 5 ```scala val propFizzBuzz15 = propFizzBuzz( `gen = genMult15`, oracle = _ => "FizzBuzz" ) ``` --- ## Multiples of 3 and 5 ```scala val propFizzBuzz15 = propFizzBuzz( gen = genMult15, `oracle = _ => "FizzBuzz"` ) ``` --- ## Key takeaways -- * you don't need to learn a new way to write tests (right away). -- * you can adapt existing tests with little work. -- * you will acquire the skills you need as you do so. --- class: center, middle name: closing # In closing --- ## If you only remember 1 slide... -- Property-based testing: * finds more flaws than example-based testing. -- * doesn't need to be harder than example-based testing. -- * importantly, is a lot of fun! --- class: center, middle name: questions # Questions? Nicolas Rinaudo • [@NicolasRinaudo] --- class: center, middle name: metamorphic # Metamorphic testing --- ## Profile moderation .center[data:image/s3,"s3://crabby-images/b7e56/b7e569d8c18d5e6656e7af02149a9230d62ce663" alt="Self"] --- ## Profile moderation .center[data:image/s3,"s3://crabby-images/c90e2/c90e220472e0687900d76fb1a3643771ee0b10bb" alt="Self"] --- ## Profile moderation .center[data:image/s3,"s3://crabby-images/c1034/c103452acdf9428e1494b99e8cf3007630d9b348" alt="Self"] --- ## Profile moderation .center[data:image/s3,"s3://crabby-images/09dc2/09dc276f8c8b20c718f8b9481b0b136ab24756d5" alt="Self"] --- ## Profile moderation .center[data:image/s3,"s3://crabby-images/45465/454658d65df0a6960b126b88d7f127d9047df858" alt="Self"] --- ## Profile moderation .center[data:image/s3,"s3://crabby-images/bd223/bd223f16870e3e085ad9b760650c243998d72e6b" alt="Self"] -- .foreground[data:image/s3,"s3://crabby-images/e16c1/e16c1fa1a1931542cef8a25b2caa5f39c9bed162" alt="Fail"] --- ## Metamorphic testing .center[data:image/s3,"s3://crabby-images/59e5d/59e5d01e06f051325b22f2fb32a7b5067bb5198e" alt="Metamorphic Testing"] --- ## Metamorphic testing .center[data:image/s3,"s3://crabby-images/3e486/3e4868e0985560b4ef2f783271d7ca87e9b0934a" alt="Metamorphic Testing"] --- ## Metamorphic testing .center[data:image/s3,"s3://crabby-images/5ec6a/5ec6a277d83d36cffbac8eb37f6a6ad9b20e1aad" alt="Metamorphic Testing"] --- ## Metamorphic testing .center[data:image/s3,"s3://crabby-images/ff1a2/ff1a24cc1301a35edf1fc574663f4451ebbbf34c" alt="Metamorphic Testing"] --- ## Metamorphic testing .center[data:image/s3,"s3://crabby-images/5c2b4/5c2b4e9c45bf886282c1e8c4ba31305f08812dd4" alt="Metamorphic Testing"] --- ## Compiler validation ```scala def fizzBuzz(i: Int) = if mult3(i) then if mult5(i) then "FizzBuzz" else "Fizz" else if mult5(i) then "Buzz" else i.toString ``` --- ## Compiler validation ```scala println(fizzBuzz(5)) ``` --- ## Compiler validation ```scala println(fizzBuzz(`5`)) ``` --- ## Compiler validation ```scala println(fizzBuzz(5)) // Buzz ``` --- ## Compiler validation ```scala println(fizzBuzz(5)) // `Buzz` ``` --- ## Compiler validation .coverage-on[ ```scala def fizzBuzz(i: Int) = * if mult3(i) then if mult5(i) then "FizzBuzz" else "Fizz" * else if mult5(i) then "Buzz" else i.toString ``` ] --- ## Compiler validation .coverage-off[ ```scala def fizzBuzz(i: Int) = if mult3(i) then * if mult5(i) then "FizzBuzz" * else "Fizz" else if mult5(i) then "Buzz" * else i.toString ``` ] --- ## Compiler validation .diff-rm[ ```scala def fizzBuzz(i: Int) = if mult3(i) then * `if mult5(i) then "FizzBuzz"` * `else "Fizz"` else if mult5(i) then "Buzz" else i.toString ``` ] --- ## Compiler validation .diff-add[ ```scala def fizzBuzz(i: Int) = * if mult3(i) then `"Fizz"` else if mult5(i) then "Buzz" else i.toString ``` ] --- ## Compiler validation .diff-rm[ ```scala def fizzBuzz(i: Int) = * if mult3(i) then `"Fizz"` else if mult5(i) then "Buzz" else i.toString ``` ] --- ## Compiler validation .diff-add[ ```scala def fizzBuzz(i: Int) = * if mult3(i) then `"Zzif"` else if mult5(i) then "Buzz" else i.toString ``` ] --- ## Compiler validation ```scala println(fizzBuzz(5)) ``` --- ## Compiler validation ```scala println(fizzBuzz(`5`)) ``` --- ## Compiler validation ```scala println(fizzBuzz(5)) // Buzz ``` --- ## Compiler validation ```scala println(fizzBuzz(5)) // `Buzz` ``` --- ## Compiler validation ```scala println(fizzBuzz(5)) // `Zzif` ``` -- .foreground[data:image/s3,"s3://crabby-images/e16c1/e16c1fa1a1931542cef8a25b2caa5f39c9bed162" alt="Fail"] --- ## Compiler validation _[Compiler Validation via Equivalence Modulo Inputs](http://vuminhle.com/pdf/pldi14-emi.pdf)_: * 79 new unique gcc bugs. * 68 new unique LLVM bugs. -- * 42 new unique scalac bugs. --- ## Key takeaways Metamorphic Testing: -- * helps generating test cases when it's too expensive. -- * helps with the oracle problem. -- * a good word to use when you want to sound smart. --- class: center, middle name: questions # Questions? Nicolas Rinaudo • [@NicolasRinaudo] [@NicolasRinaudo]:https://twitter.com/NicolasRinaudo