diff options
author | Eduardo Julian | 2019-03-24 01:00:56 -0400 |
---|---|---|
committer | Eduardo Julian | 2019-03-24 01:00:56 -0400 |
commit | 711c573bef4b7a6d809568ebfc196a7f8688307f (patch) | |
tree | 06d2287e2b5e80608c7f47a8c5eb664d0462d146 /stdlib/source/test | |
parent | 4c6d153acdbfa7e71109dbe9c1ae1fcca9e914e6 (diff) |
Ported tests for collection-related modules.
Diffstat (limited to '')
20 files changed, 1243 insertions, 1208 deletions
diff --git a/stdlib/source/test/lux.lux b/stdlib/source/test/lux.lux index 3855f350f..781f3edde 100644 --- a/stdlib/source/test/lux.lux +++ b/stdlib/source/test/lux.lux @@ -120,24 +120,6 @@ ## [concurrency ## ## [semaphore (#+)] ## ]] - ## [data - ## ## [collection - ## ## [array (#+)] - ## ## [bits (#+)] - ## ## [list (#+)] - ## ## [stack (#+)] - ## ## [row (#+)] - ## ## [sequence (#+)] - ## ## [dictionary (#+) - ## ## ["dictionary_." ordered]] - ## ## [set (#+) - ## ## ["set_." ordered]] - ## ## [queue (#+) - ## ## [priority (#+)]] - ## ## [tree - ## ## [rose (#+) - ## ## [zipper (#+)]]]] - ## ] ## [math (#+) ## [random (#+)] ## [modular (#+)] diff --git a/stdlib/source/test/lux/control/fold.lux b/stdlib/source/test/lux/control/fold.lux new file mode 100644 index 000000000..7d7ea8d83 --- /dev/null +++ b/stdlib/source/test/lux/control/fold.lux @@ -0,0 +1,21 @@ +(.module: + [lux #* + ["_" test (#+ Test)] + data/text/format + ["r" math/random] + [control + [monad (#+ do)]]] + [// + [functor (#+ Injection Comparison)]] + {1 + ["." / (#+ Fold)]}) + +(def: #export (spec injection comparison (^open "/@.")) + (All [f] (-> (Injection f) (Comparison f) (Fold f) Test)) + (_.context (%name (name-of /.Fold)) + (do r.monad + [subject r.nat + parameter r.nat] + (_.test "Can fold." + (n/= (/@fold n/+ parameter (injection subject)) + (n/+ parameter subject)))))) diff --git a/stdlib/source/test/lux/control/functor.lux b/stdlib/source/test/lux/control/functor.lux index a8fbfa6fc..68c8db55b 100644 --- a/stdlib/source/test/lux/control/functor.lux +++ b/stdlib/source/test/lux/control/functor.lux @@ -1,14 +1,12 @@ (.module: [lux #* + ["_" test (#+ Test)] + data/text/format + ["r" math/random] [control + [equivalence (#+ Equivalence)] [monad (#+ do)]] - [data - [text - format]] - ["." function] - [math - ["r" random]] - ["_" test (#+ Test)]] + ["." function]] {1 ["." / (#+ Functor)]}) @@ -17,29 +15,29 @@ (type: #export (Comparison f) (All [a] - (-> (-> a a Bit) - (-> (f a) (f a) Bit)))) + (-> (Equivalence a) + (Equivalence (f a))))) -(def: (identity injection comparison (^open "_;.")) +(def: (identity injection comparison (^open "/@.")) (All [f] (-> (Injection f) (Comparison f) (Functor f) Test)) (do r.monad [sample (:: @ map injection r.nat)] (_.test "Identity." ((comparison n/=) - (_;map function.identity sample) + (/@map function.identity sample) sample)))) -(def: (homomorphism injection comparison (^open "_;.")) +(def: (homomorphism injection comparison (^open "/@.")) (All [f] (-> (Injection f) (Comparison f) (Functor f) Test)) (do r.monad [sample r.nat increase (:: @ map n/+ r.nat)] (_.test "Homomorphism." ((comparison n/=) - (_;map increase (injection sample)) + (/@map increase (injection sample)) (injection (increase sample)))))) -(def: (composition injection comparison (^open "_;.")) +(def: (composition injection comparison (^open "/@.")) (All [f] (-> (Injection f) (Comparison f) (Functor f) Test)) (do r.monad [sample (:: @ map injection r.nat) @@ -47,8 +45,8 @@ decrease (:: @ map n/- r.nat)] (_.test "Composition." ((comparison n/=) - (|> sample (_;map increase) (_;map decrease)) - (|> sample (_;map (|>> increase decrease))))))) + (|> sample (/@map increase) (/@map decrease)) + (|> sample (/@map (|>> increase decrease))))))) (def: #export (spec injection comparison functor) (All [f] (-> (Injection f) (Comparison f) (Functor f) Test)) diff --git a/stdlib/source/test/lux/data.lux b/stdlib/source/test/lux/data.lux index 9175d970e..92680f03a 100644 --- a/stdlib/source/test/lux/data.lux +++ b/stdlib/source/test/lux/data.lux @@ -24,7 +24,8 @@ ["#/." regex]] [format ["#." json] - ["#." xml]]]) + ["#." xml]] + ["#." collection]]) (def: number Test @@ -66,4 +67,5 @@ ..number ..text ..format + /collection.test )) diff --git a/stdlib/source/test/lux/data/collection.lux b/stdlib/source/test/lux/data/collection.lux new file mode 100644 index 000000000..ad86d3225 --- /dev/null +++ b/stdlib/source/test/lux/data/collection.lux @@ -0,0 +1,62 @@ +(.module: + [lux #* + ["_" test (#+ Test)]] + ["." / #_ + ["#." array] + ["#." bits] + ["#." list] + ["#." row] + ["#." sequence] + ["#." stack] + ["#." dictionary + ["#/." ordered]] + ["#." queue + ["#/." priority]] + ["#." set + ["#/." ordered]] + ["#." tree #_ + ["#/." rose + ["#/." zipper]]]]) + +(def: dictionary + Test + ($_ _.and + /dictionary.test + /dictionary/ordered.test + )) + +(def: queue + Test + ($_ _.and + /queue.test + /queue/priority.test + )) + +(def: set + Test + ($_ _.and + /set.test + /set/ordered.test + )) + +(def: tree + Test + ($_ _.and + /tree/rose.test + /tree/rose/zipper.test + )) + +(def: #export test + Test + ($_ _.and + /array.test + /bits.test + /list.test + /row.test + /sequence.test + /stack.test + ..dictionary + ..queue + ..set + ..tree + )) diff --git a/stdlib/source/test/lux/data/collection/array.lux b/stdlib/source/test/lux/data/collection/array.lux index 47c384cb7..8814a6e88 100644 --- a/stdlib/source/test/lux/data/collection/array.lux +++ b/stdlib/source/test/lux/data/collection/array.lux @@ -1,143 +1,117 @@ (.module: [lux #* + data/text/format + ["_" test (#+ Test)] [control + pipe [monad (#+ do Monad)] - pipe] + {[0 #test] + [/ + ["$." equivalence] + ["$." monoid] + ["$." fold] + ["$." functor (#+ Injection)]]}] [data - ["." number] ["." maybe] + [number + ["." nat]] [collection - ["@" array (#+ Array)] ["." list]]] [math - ["r" random]]] - lux/test) + ["r" random (#+ Random)]]] + {1 + ["." / (#+ Array)]}) + +(def: injection + (Injection Array) + (|>> list /.from-list)) (def: bounded-size - (r.Random Nat) + (Random Nat) (|> r.nat (:: r.monad map (|>> (n/% 100) (n/+ 1))))) -(context: "Arrays and their copies" - (<| (times 100) - (do @ - [size bounded-size - original (r.array size r.nat) - #let [clone (@.clone original) - copy (: (Array Nat) - (@.new size)) - manual-copy (: (Array Nat) - (@.new size))]] - ($_ seq - (test "Size function must correctly return size of array." - (n/= size (@.size original))) - (test "Cloning an array should yield and identical array, but not the same one." - (and (:: (@.equivalence number.equivalence) = original clone) - (not (is? original clone)))) - (test "Full-range manual copies should give the same result as cloning." - (exec (@.copy size 0 original 0 copy) - (and (:: (@.equivalence number.equivalence) = original copy) - (not (is? original copy))))) - (test "Array folding should go over all values." - (exec (:: @.fold fold - (function (_ x idx) - (exec (@.write idx x manual-copy) - (inc idx))) - 0 - original) - (:: (@.equivalence number.equivalence) = original manual-copy))) - (test "Transformations between (full) arrays and lists shouldn't cause lose or change any values." - (|> original - @.to-list @.from-list - (:: (@.equivalence number.equivalence) = original))) - )))) - -(context: "Array mutation" - (<| (times 100) - (do @ - [size bounded-size - idx (:: @ map (n/% size) r.nat) - array (|> (r.array size r.nat) - (r.filter (|>> @.to-list (list.any? n/odd?)))) - #let [value (maybe.assume (@.read idx array))]] - ($_ seq - (test "Shouldn't be able to find a value in an unoccupied cell." - (case (@.read idx (@.delete idx array)) - (#.Some _) #0 - #.None #1)) - (test "You should be able to access values put into the array." - (case (@.read idx (@.write idx value array)) - (#.Some value') (n/= value' value) - #.None #0)) - (test "All cells should be occupied on a full array." - (and (n/= size (@.occupied array)) - (n/= 0 (@.vacant array)))) - (test "Filtering mutates the array to remove invalid values." - (exec (@.filter! n/even? array) - (and (n/< size (@.occupied array)) - (n/> 0 (@.vacant array)) - (n/= size (n/+ (@.occupied array) - (@.vacant array)))))) - )))) - -(context: "Finding values." - (<| (times 100) - (do @ - [size bounded-size - array (|> (r.array size r.nat) - (r.filter (|>> @.to-list (list.any? n/even?))))] - ($_ seq - (test "Can find values inside arrays." - (|> (@.find n/even? array) - (case> (#.Some _) #1 - #.None #0))) - (test "Can find values inside arrays (with access to indices)." - (|> (@.find+ (function (_ idx n) - (and (n/even? n) - (n/< size idx))) - array) - (case> (#.Some _) #1 - #.None #0))))))) - -(context: "Functor" - (<| (times 100) - (do @ - [size bounded-size - array (r.array size r.nat)] - (let [(^open ".") @.functor - (^open ".") (@.equivalence number.equivalence)] - ($_ seq - (test "Functor shouldn't alter original array." - (let [copy (map id array)] - (and (= array copy) - (not (is? array copy))))) - (test "Functor should go over all available array elements." - (let [there (map inc array) - back-again (map dec there)] - (and (not (= array there)) - (= array back-again))))))))) +(def: #export test + Test + (<| (_.context (%name (name-of /.Array))) + (do r.monad + [size bounded-size] + ($_ _.and + ($equivalence.spec (/.equivalence nat.equivalence) (r.array size r.nat)) + ($monoid.spec (/.equivalence nat.equivalence) /.monoid (r.array size r.nat)) + ($functor.spec ..injection /.equivalence /.functor) + ($fold.spec ..injection /.equivalence /.fold) -(context: "Monoid" - (<| (times 100) - (do @ - [sizeL bounded-size - sizeR bounded-size - left (r.array sizeL r.nat) - right (r.array sizeR r.nat) - #let [(^open ".") @.monoid - (^open ".") (@.equivalence number.equivalence) - fusion (compose left right)]] - ($_ seq - (test "Appending two arrays should produce a new one twice as large." - (n/= (n/+ sizeL sizeR) (@.size fusion))) - (test "First elements of fused array should equal the first array." - (|> (: (Array Nat) - (@.new sizeL)) - (@.copy sizeL 0 fusion 0) - (= left))) - (test "Last elements of fused array should equal the second array." - (|> (: (Array Nat) - (@.new sizeR)) - (@.copy sizeR sizeL fusion 0) - (= right))) + (do r.monad + [size bounded-size + original (r.array size r.nat)] + ($_ _.and + (_.test "Size function must correctly return size of array." + (n/= size (/.size original))) + (_.test "Cloning an array should yield and identical array, but not the same one." + (let [clone (/.clone original)] + (and (:: (/.equivalence nat.equivalence) = original clone) + (not (is? original clone))))) + (_.test "Full-range manual copies should give the same result as cloning." + (let [copy (: (Array Nat) + (/.new size))] + (exec (/.copy size 0 original 0 copy) + (and (:: (/.equivalence nat.equivalence) = original copy) + (not (is? original copy)))))) + (_.test "Array folding should go over all values." + (let [manual-copy (: (Array Nat) + (/.new size))] + (exec (:: /.fold fold + (function (_ x idx) + (exec (/.write idx x manual-copy) + (inc idx))) + 0 + original) + (:: (/.equivalence nat.equivalence) = original manual-copy)))) + (_.test "Transformations between (full) arrays and lists shouldn't cause lose or change any values." + (|> original + /.to-list /.from-list + (:: (/.equivalence nat.equivalence) = original))) + )) + (do r.monad + [size bounded-size + idx (:: @ map (n/% size) r.nat) + array (|> (r.array size r.nat) + (r.filter (|>> /.to-list (list.any? n/odd?)))) + #let [value (maybe.assume (/.read idx array))]] + ($_ _.and + (_.test "Shouldn't be able to find a value in an unoccupied cell." + (case (/.read idx (/.delete idx array)) + (#.Some _) false + #.None true)) + (_.test "You should be able to access values put into the array." + (case (/.read idx (/.write idx value array)) + (#.Some value') (n/= value' value) + #.None false)) + (_.test "All cells should be occupied on a full array." + (and (n/= size (/.occupied array)) + (n/= 0 (/.vacant array)))) + (_.test "Filtering mutates the array to remove invalid values." + (exec (/.filter! n/even? array) + (and (n/< size (/.occupied array)) + (n/> 0 (/.vacant array)) + (n/= size (n/+ (/.occupied array) + (/.vacant array)))))) + )) + (do r.monad + [size bounded-size + array (|> (r.array size r.nat) + (r.filter (|>> /.to-list (list.any? n/even?))))] + ($_ _.and + (_.test "Can find values inside arrays." + (|> (/.find n/even? array) + (case> (#.Some _) true + #.None false))) + (_.test "Can find values inside arrays (with access to indices)." + (|> (/.find+ (function (_ idx n) + (and (n/even? n) + (n/< size idx))) + array) + (case> (#.Some _) true + #.None false))))) )))) diff --git a/stdlib/source/test/lux/data/collection/bits.lux b/stdlib/source/test/lux/data/collection/bits.lux index e932aacef..ac7e983f9 100644 --- a/stdlib/source/test/lux/data/collection/bits.lux +++ b/stdlib/source/test/lux/data/collection/bits.lux @@ -1,87 +1,82 @@ (.module: [lux #* + data/text/format + ["_" test (#+ Test)] [control [monad (#+ do)] - ["." predicate]] - [data - [collection - ["/" bits]]] + ["." predicate] + {[0 #test] + [/ + ["$." equivalence]]}] [math - ["r" random]]] - lux/test - [test - [lux - [control - ["_eq" equivalence]]]]) + ["r" random (#+ Random)]]] + {1 + ["." / (#+ Bits)]}) (def: (size min max) - (-> Nat Nat (r.Random Nat)) + (-> Nat Nat (Random Nat)) (|> r.nat (:: r.monad map (|>> (n/% max) (n/max min))))) -(def: bits - (r.Random /.Bits) +(def: #export bits + (Random Bits) (do r.monad [size (size 1 1,000) idx (|> r.nat (:: @ map (n/% size)))] (wrap (|> /.empty (/.set idx))))) -(context: "Bits." - (<| (times 100) - (do @ - [size (size 1 1,000) - idx (|> r.nat (:: @ map (n/% size))) - sample bits] - ($_ seq - (test "Can set individual bits." - (and (|> /.empty (/.get idx) not) - (|> /.empty (/.set idx) (/.get idx)))) - (test "Can clear individual bits." - (|> /.empty (/.set idx) (/.clear idx) (/.get idx) not)) - (test "Can flip individual bits." - (and (|> /.empty (/.flip idx) (/.get idx)) - (|> /.empty (/.flip idx) (/.flip idx) (/.get idx) not))) - - (test "Bits (only) grow when (and as much as) necessary." - (and (n/= 0 (/.capacity /.empty)) - (|> /.empty (/.set idx) /.capacity - (n/- idx) - (predicate.union (n/>= 0) - (n/< /.chunk-size))))) - (test "Bits (must) shrink when (and as much as) possible." - (let [grown (/.flip idx /.empty)] - (and (n/> 0 (/.capacity grown)) - (is? /.empty (/.flip idx grown))))) - - (test "Intersection can be detected when there are set bits in common." - (and (not (/.intersects? /.empty - /.empty)) - (/.intersects? (/.set idx /.empty) - (/.set idx /.empty)) - (not (/.intersects? (/.set (inc idx) /.empty) - (/.set idx /.empty))))) - (test "Cannot intersect with one's opposite." - (not (/.intersects? sample (/.not sample)))) - - (test "'and' with oneself changes nothing" - (:: /.equivalence = sample (/.and sample sample))) - (test "'and' with one's opposite yields the empty bit-set." - (is? /.empty (/.and sample (/.not sample)))) - - (test "'or' with one's opposite fully saturates a bit-set." - (n/= (/.size (/.or sample (/.not sample))) - (/.capacity sample))) - - (test "'xor' with oneself yields the empty bit-set." - (is? /.empty (/.xor sample sample))) - (test "'xor' with one's opposite fully saturates a bit-set." - (n/= (/.size (/.xor sample (/.not sample))) - (/.capacity sample))) - - (test "Double negation results in original bit-set." - (:: /.equivalence = sample (/.not (/.not sample)))) - (test "Negation does not affect the empty bit-set." - (is? /.empty (/.not /.empty))) - - (_eq.spec /.equivalence ..bits) - )))) +(def: #export test + Test + (<| (_.context (%name (name-of /.Bits))) + ($_ _.and + ($equivalence.spec /.equivalence ..bits) + (do r.monad + [size (size 1 1,000) + idx (|> r.nat (:: @ map (n/% size))) + sample bits] + ($_ _.and + (_.test "Can set individual bits." + (and (|> /.empty (/.get idx) not) + (|> /.empty (/.set idx) (/.get idx)))) + (_.test "Can clear individual bits." + (|> /.empty (/.set idx) (/.clear idx) (/.get idx) not)) + (_.test "Can flip individual bits." + (and (|> /.empty (/.flip idx) (/.get idx)) + (|> /.empty (/.flip idx) (/.flip idx) (/.get idx) not))) + (_.test "Bits (only) grow when (and as much as) necessary." + (and (n/= 0 (/.capacity /.empty)) + (|> /.empty (/.set idx) /.capacity + (n/- idx) + (predicate.union (n/>= 0) + (n/< /.chunk-size))))) + (_.test "Bits (must) shrink when (and as much as) possible." + (let [grown (/.flip idx /.empty)] + (and (n/> 0 (/.capacity grown)) + (is? /.empty (/.flip idx grown))))) + (_.test "Intersection can be detected when there are set bits in common." + (and (not (/.intersects? /.empty + /.empty)) + (/.intersects? (/.set idx /.empty) + (/.set idx /.empty)) + (not (/.intersects? (/.set (inc idx) /.empty) + (/.set idx /.empty))))) + (_.test "Cannot intersect with one's opposite." + (not (/.intersects? sample (/.not sample)))) + (_.test "'and' with oneself changes nothing" + (:: /.equivalence = sample (/.and sample sample))) + (_.test "'and' with one's opposite yields the empty bit-set." + (is? /.empty (/.and sample (/.not sample)))) + + (_.test "'or' with one's opposite fully saturates a bit-set." + (n/= (/.size (/.or sample (/.not sample))) + (/.capacity sample))) + (_.test "'xor' with oneself yields the empty bit-set." + (is? /.empty (/.xor sample sample))) + (_.test "'xor' with one's opposite fully saturates a bit-set." + (n/= (/.size (/.xor sample (/.not sample))) + (/.capacity sample))) + (_.test "Double negation results in original bit-set." + (:: /.equivalence = sample (/.not (/.not sample)))) + (_.test "Negation does not affect the empty bit-set." + (is? /.empty (/.not /.empty))) + ))))) diff --git a/stdlib/source/test/lux/data/collection/dictionary.lux b/stdlib/source/test/lux/data/collection/dictionary.lux index 80d673574..e559a2453 100644 --- a/stdlib/source/test/lux/data/collection/dictionary.lux +++ b/stdlib/source/test/lux/data/collection/dictionary.lux @@ -1,129 +1,129 @@ (.module: [lux #* + data/text/format + ["_" test (#+ Test)] [control - [monad (#+ do Monad)] - ["eq" equivalence]] + [monad (#+ do)] + ["eq" equivalence] + {[0 #test] + [/ + ["$." equivalence] + ["$." functor (#+ Injection)]]}] [data - ["." number] ["." maybe] + [number + ["." nat]] [collection - ["&" dictionary] - ["." list ("#;." functor)]]] + ["." list ("#@." functor)]]] [math ["r" random]]] - lux/test) + {1 + ["." /]}) -(context: "Dictionaries." - (<| (times 100) - (do @ +(def: injection + (Injection (/.Dictionary Nat)) + (|>> [0] list (/.from-list nat.hash))) + +(def: #export test + Test + (<| (_.context (%name (name-of /.Dictionary))) + (do r.monad [#let [capped-nat (:: r.monad map (n/% 100) r.nat)] size capped-nat - dict (r.dictionary number.hash size r.nat capped-nat) - non-key (|> r.nat (r.filter (function (_ key) (not (&.contains? key dict))))) - test-val (|> r.nat (r.filter (function (_ val) (not (list.member? number.equivalence (&.values dict) val)))))] - ($_ seq - (test "Size function should correctly represent Dictionary size." - (n/= size (&.size dict))) - - (test "Dictionaries of size 0 should be considered empty." - (if (n/= 0 size) - (&.empty? dict) - (not (&.empty? dict)))) - - (test "The functions 'entries', 'keys' and 'values' should be synchronized." - (:: (list.equivalence (eq.product number.equivalence number.equivalence)) = - (&.entries dict) - (list.zip2 (&.keys dict) - (&.values dict)))) - - (test "Dictionary should be able to recognize it's own keys." - (list.every? (function (_ key) (&.contains? key dict)) - (&.keys dict))) - - (test "Should be able to get every key." - (list.every? (function (_ key) (case (&.get key dict) - (#.Some _) #1 - _ #0)) - (&.keys dict))) - - (test "Shouldn't be able to access non-existant keys." - (case (&.get non-key dict) - (#.Some _) #0 - _ #1)) - - (test "Should be able to put and then get a value." - (case (&.get non-key (&.put non-key test-val dict)) - (#.Some v) (n/= test-val v) - _ #1)) - - (test "Should be able to put~ and then get a value." - (case (&.get non-key (&.put~ non-key test-val dict)) - (#.Some v) (n/= test-val v) - _ #1)) + dict (r.dictionary nat.hash size r.nat capped-nat) + non-key (|> r.nat (r.filter (function (_ key) (not (/.contains? key dict))))) + test-val (|> r.nat (r.filter (function (_ val) (not (list.member? nat.equivalence (/.values dict) val)))))] + ($_ _.and + ($equivalence.spec (/.equivalence nat.equivalence) + (r.dictionary nat.hash size r.nat r.nat)) + ($functor.spec ..injection /.equivalence /.functor) - (test "Shouldn't be able to put~ an existing key." - (or (n/= 0 size) - (let [first-key (|> dict &.keys list.head maybe.assume)] - (case (&.get first-key (&.put~ first-key test-val dict)) - (#.Some v) (not (n/= test-val v)) - _ #1)))) - - (test "Removing a key should make it's value inaccessible." - (let [base (&.put non-key test-val dict)] - (and (&.contains? non-key base) - (not (&.contains? non-key (&.remove non-key base)))))) - - (test "Should be possible to update values via their keys." - (let [base (&.put non-key test-val dict) - updt (&.update non-key inc base)] - (case [(&.get non-key base) (&.get non-key updt)] - [(#.Some x) (#.Some y)] - (n/= (inc x) y) - - _ - #0))) - - (test "Additions and removals to a Dictionary should affect its size." - (let [plus (&.put non-key test-val dict) - base (&.remove non-key plus)] - (and (n/= (inc (&.size dict)) (&.size plus)) - (n/= (dec (&.size plus)) (&.size base))))) - - (test "A Dictionary should equal itself & going to<->from lists shouldn't change that." - (let [(^open ".") (&.equivalence number.equivalence)] - (and (= dict dict) - (|> dict &.entries (&.from-list number.hash) (= dict))))) - - (test "Merging a Dictionary to itself changes nothing." - (let [(^open ".") (&.equivalence number.equivalence)] - (= dict (&.merge dict dict)))) - - (test "If you merge, and the second dict has overlapping keys, it should overwrite yours." - (let [dict' (|> dict &.entries - (list;map (function (_ [k v]) [k (inc v)])) - (&.from-list number.hash)) - (^open ".") (&.equivalence number.equivalence)] - (= dict' (&.merge dict' dict)))) - - (test "Can merge values in such a way that they become combined." - (list.every? (function (_ [x x*2]) (n/= (n/* 2 x) x*2)) - (list.zip2 (&.values dict) - (&.values (&.merge-with n/+ dict dict))))) + (_.test "Size function should correctly represent Dictionary size." + (n/= size (/.size dict))) + (_.test "Dictionaries of size 0 should be considered empty." + (if (n/= 0 size) + (/.empty? dict) + (not (/.empty? dict)))) + (_.test "The functions 'entries', 'keys' and 'values' should be synchronized." + (:: (list.equivalence (eq.product nat.equivalence nat.equivalence)) = + (/.entries dict) + (list.zip2 (/.keys dict) + (/.values dict)))) + (_.test "Dictionary should be able to recognize it's own keys." + (list.every? (function (_ key) (/.contains? key dict)) + (/.keys dict))) + (_.test "Should be able to get every key." + (list.every? (function (_ key) (case (/.get key dict) + (#.Some _) #1 + _ #0)) + (/.keys dict))) + (_.test "Shouldn't be able to access non-existant keys." + (case (/.get non-key dict) + (#.Some _) #0 + _ #1)) + (_.test "Should be able to put and then get a value." + (case (/.get non-key (/.put non-key test-val dict)) + (#.Some v) (n/= test-val v) + _ #1)) - (test "Should be able to select subset of keys from dict." - (|> dict - (&.put non-key test-val) - (&.select (list non-key)) - &.size - (n/= 1))) + (_.test "Should be able to try-put and then get a value." + (case (/.get non-key (/.try-put non-key test-val dict)) + (#.Some v) (n/= test-val v) + _ #1)) + (_.test "Shouldn't be able to try-put an existing key." + (or (n/= 0 size) + (let [first-key (|> dict /.keys list.head maybe.assume)] + (case (/.get first-key (/.try-put first-key test-val dict)) + (#.Some v) (not (n/= test-val v)) + _ #1)))) + (_.test "Removing a key should make it's value inaccessible." + (let [base (/.put non-key test-val dict)] + (and (/.contains? non-key base) + (not (/.contains? non-key (/.remove non-key base)))))) + (_.test "Should be possible to update values via their keys." + (let [base (/.put non-key test-val dict) + updt (/.update non-key inc base)] + (case [(/.get non-key base) (/.get non-key updt)] + [(#.Some x) (#.Some y)] + (n/= (inc x) y) - (test "Should be able to re-bind existing values to different keys." - (or (n/= 0 size) - (let [first-key (|> dict &.keys list.head maybe.assume) - rebound (&.re-bind first-key non-key dict)] - (and (n/= (&.size dict) (&.size rebound)) - (&.contains? non-key rebound) - (not (&.contains? first-key rebound)) - (n/= (maybe.assume (&.get first-key dict)) - (maybe.assume (&.get non-key rebound))))))) + _ + #0))) + (_.test "Additions and removals to a Dictionary should affect its size." + (let [plus (/.put non-key test-val dict) + base (/.remove non-key plus)] + (and (n/= (inc (/.size dict)) (/.size plus)) + (n/= (dec (/.size plus)) (/.size base))))) + (_.test "A Dictionary should equal itself & going to<->from lists shouldn't change that." + (let [(^open ".") (/.equivalence nat.equivalence)] + (and (= dict dict) + (|> dict /.entries (/.from-list nat.hash) (= dict))))) + (_.test "Merging a Dictionary to itself changes nothing." + (let [(^open ".") (/.equivalence nat.equivalence)] + (= dict (/.merge dict dict)))) + (_.test "If you merge, and the second dict has overlapping keys, it should overwrite yours." + (let [dict' (|> dict /.entries + (list@map (function (_ [k v]) [k (inc v)])) + (/.from-list nat.hash)) + (^open ".") (/.equivalence nat.equivalence)] + (= dict' (/.merge dict' dict)))) + (_.test "Can merge values in such a way that they become combined." + (list.every? (function (_ [x x*2]) (n/= (n/* 2 x) x*2)) + (list.zip2 (/.values dict) + (/.values (/.merge-with n/+ dict dict))))) + (_.test "Should be able to select subset of keys from dict." + (|> dict + (/.put non-key test-val) + (/.select (list non-key)) + /.size + (n/= 1))) + (_.test "Should be able to re-bind existing values to different keys." + (or (n/= 0 size) + (let [first-key (|> dict /.keys list.head maybe.assume) + rebound (/.re-bind first-key non-key dict)] + (and (n/= (/.size dict) (/.size rebound)) + (/.contains? non-key rebound) + (not (/.contains? first-key rebound)) + (n/= (maybe.assume (/.get first-key dict)) + (maybe.assume (/.get non-key rebound))))))) )))) diff --git a/stdlib/source/test/lux/data/collection/dictionary/ordered.lux b/stdlib/source/test/lux/data/collection/dictionary/ordered.lux index 2d1f5a0ba..4cd7880ba 100644 --- a/stdlib/source/test/lux/data/collection/dictionary/ordered.lux +++ b/stdlib/source/test/lux/data/collection/dictionary/ordered.lux @@ -1,91 +1,110 @@ (.module: [lux #* + data/text/format + ["_" test (#+ Test)] [control - [monad (#+ Monad do)] - [equivalence (#+ Equivalence)]] + [monad (#+ do)] + [equivalence (#+ Equivalence)] + [order (#+ Order)] + {[0 #test] + [/ + ["$." equivalence]]}] [data ["." product] - ["." number] + [number + ["." nat]] [collection - ["s" set] - ["dict" dictionary - ["&" ordered]] - ["." list ("#;." functor)]]] + ["." set] + ["." list ("#@." functor)]]] [math - ["r" random]]] - lux/test) + ["r" random (#+ Random) ("#@." monad)]]] + {1 + ["." /]}) -(context: "Dictionary" - (<| (times 100) - (do @ +(def: #export (dictionary order gen-key gen-value size) + (All [k v] + (-> (Order k) (Random k) (Random v) Nat (Random (/.Dictionary k v)))) + (case size + 0 + (r@wrap (/.new order)) + + _ + (do r.monad + [partial (dictionary order gen-key gen-value (dec size)) + key (r.filter (function (_ candidate) + (not (/.contains? candidate partial))) + gen-key) + value gen-value] + (wrap (/.put key value partial))))) + +(def: #export test + Test + (<| (_.context (%name (name-of /.Dictionary))) + (do r.monad [size (|> r.nat (:: @ map (n/% 100))) - keys (r.set number.nat-hash size r.nat) - values (r.set number.nat-hash size r.nat) - extra-key (|> r.nat (r.filter (|>> (s.member? keys) not))) + keys (r.set nat.hash size r.nat) + values (r.set nat.hash size r.nat) + extra-key (|> r.nat (r.filter (|>> (set.member? keys) not))) extra-value r.nat - #let [pairs (list.zip2 (s.to-list keys) - (s.to-list values)) - sample (&.from-list number.nat-order pairs) + #let [pairs (list.zip2 (set.to-list keys) + (set.to-list values)) + sample (/.from-list nat.order pairs) sorted-pairs (list.sort (function (_ [left _] [right _]) (n/< left right)) pairs) - sorted-values (list;map product.right sorted-pairs) - (^open "&;.") (&.equivalence number.nat-equivalence)]] - ($_ seq - (test "Can query the size of a dictionary." - (n/= size (&.size sample))) - - (test "Can query value for minimum key." - (case [(&.min sample) (list.head sorted-values)] - [#.None #.None] - #1 - - [(#.Some reference) (#.Some sample)] - (n/= reference sample) - - _ - #0)) - - (test "Can query value for maximum key." - (case [(&.max sample) (list.last sorted-values)] - [#.None #.None] - #1 - - [(#.Some reference) (#.Some sample)] - (n/= reference sample) - - _ - #0)) + sorted-values (list@map product.right sorted-pairs) + (^open "/@.") (/.equivalence nat.equivalence)]] + ($_ _.and + ($equivalence.spec (/.equivalence nat.equivalence) (..dictionary nat.order r.nat r.nat size)) + + (_.test "Can query the size of a dictionary." + (n/= size (/.size sample))) + (_.test "Can query value for minimum key." + (case [(/.min sample) (list.head sorted-values)] + [#.None #.None] + #1 - (test "Converting dictionaries to/from lists cannot change their values." - (|> sample - &.entries (&.from-list number.nat-order) - (&;= sample))) + [(#.Some reference) (#.Some sample)] + (n/= reference sample) - (test "Order is preserved." - (let [(^open "list;.") (list.equivalence (: (Equivalence [Nat Nat]) - (function (_ [kr vr] [ks vs]) - (and (n/= kr ks) - (n/= vr vs)))))] - (list;= (&.entries sample) - sorted-pairs))) + _ + #0)) + (_.test "Can query value for maximum key." + (case [(/.max sample) (list.last sorted-values)] + [#.None #.None] + #1 - (test "Every key in a dictionary must be identifiable." - (list.every? (function (_ key) (&.contains? key sample)) - (&.keys sample))) + [(#.Some reference) (#.Some sample)] + (n/= reference sample) - (test "Can add and remove elements in a dictionary." - (and (not (&.contains? extra-key sample)) - (let [sample' (&.put extra-key extra-value sample) - sample'' (&.remove extra-key sample')] - (and (&.contains? extra-key sample') - (not (&.contains? extra-key sample'')) - (case [(&.get extra-key sample') - (&.get extra-key sample'')] - [(#.Some found) #.None] - (n/= extra-value found) + _ + #0)) + (_.test "Converting dictionaries to/from lists cannot change their values." + (|> sample + /.entries (/.from-list nat.order) + (/@= sample))) + (_.test "Order is preserved." + (let [(^open "list@.") (list.equivalence (: (Equivalence [Nat Nat]) + (function (_ [kr vr] [ks vs]) + (and (n/= kr ks) + (n/= vr vs)))))] + (list@= (/.entries sample) + sorted-pairs))) + (_.test "Every key in a dictionary must be identifiable." + (list.every? (function (_ key) (/.contains? key sample)) + (/.keys sample))) + (_.test "Can add and remove elements in a dictionary." + (and (not (/.contains? extra-key sample)) + (let [sample' (/.put extra-key extra-value sample) + sample'' (/.remove extra-key sample')] + (and (/.contains? extra-key sample') + (not (/.contains? extra-key sample'')) + (case [(/.get extra-key sample') + (/.get extra-key sample'')] + [(#.Some found) #.None] + (n/= extra-value found) - _ - #0))) - )) + _ + #0))) + )) )))) diff --git a/stdlib/source/test/lux/data/collection/list.lux b/stdlib/source/test/lux/data/collection/list.lux index e5ec2b5b2..958025e8b 100644 --- a/stdlib/source/test/lux/data/collection/list.lux +++ b/stdlib/source/test/lux/data/collection/list.lux @@ -1,239 +1,193 @@ (.module: [lux #* + data/text/format + ["_" test (#+ Test)] ["." io] [control - [monad (#+ do Monad)] - pipe] + pipe + [monad (#+ do)] + {[0 #test] + [/ + ["$." equivalence] + ["$." monoid] + ["$." fold] + ["$." functor] + ["$." apply] + ["$." monad]]}] [data - ["." number] ["." bit] ["." product] ["." maybe] - [collection - ["&" list]]] + [number + ["." nat] + ["." int]]] [math ["r" random]]] - lux/test) + {1 + ["." / ("#@." monad)]}) (def: bounded-size (r.Random Nat) (|> r.nat (:: r.monad map (|>> (n/% 100) (n/+ 10))))) -(context: "Lists: Part 1" - (<| (times 100) - (do @ +(def: signatures + Test + (do r.monad + [size bounded-size] + ($_ _.and + ($equivalence.spec (/.equivalence nat.equivalence) (r.list size r.nat)) + ($monoid.spec (/.equivalence nat.equivalence) /.monoid (r.list size r.nat)) + ($fold.spec /@wrap /.equivalence /.fold) + ($functor.spec /@wrap /.equivalence /.functor) + ($apply.spec /@wrap /.equivalence /.apply) + ($monad.spec /@wrap /.equivalence /.monad) + + (do @ + [parameter r.nat + subject r.nat] + (let [lift (/.lift io.monad) + (^open "io@.") io.monad + expected (n/+ parameter subject)] + (_.test "Can add list functionality to any monad." + (|> (io.run (do (/.with io.monad) + [a (lift (io@wrap parameter)) + b (wrap subject)] + (wrap (n/+ a b)))) + (case> (^ (list actual)) + (n/= expected actual) + + _ + false))))) + ))) + +(def: #export test + Test + (<| (_.context (%name (name-of .List))) + (do r.monad [size bounded-size + #let [(^open "/@.") (/.equivalence nat.equivalence) + (^open "/@.") /.functor + (^open "/@.") /.monoid] idx (:: @ map (n/% size) r.nat) sample (r.list size r.nat) other-size bounded-size other-sample (r.list other-size r.nat) - separator r.nat - #let [(^open ".") (&.equivalence number.equivalence) - (^open "&;.") &.functor]] - ($_ seq - (test "The size function should correctly portray the size of the list." - (n/= size (&.size sample))) - - (test "The repeat function should produce as many elements as asked of it." - (n/= size (&.size (&.repeat size [])))) - - (test "Reversing a list does not change it's size." - (n/= (&.size sample) - (&.size (&.reverse sample)))) + separator r.nat] + ($_ _.and + ..signatures - (test "Reversing a list twice results in the original list." - (= sample - (&.reverse (&.reverse sample)))) - - (test "Filtering by a predicate and its complement should result in a number of elements equal to the original list." - (and (n/= (&.size sample) - (n/+ (&.size (&.filter n/even? sample)) - (&.size (&.filter (bit.complement n/even?) sample)))) - (let [[plus minus] (&.partition n/even? sample)] - (n/= (&.size sample) - (n/+ (&.size plus) - (&.size minus)))))) - - (test "If every element in a list satisfies a predicate, there can't be any that satisfy its complement." - (if (&.every? n/even? sample) - (and (not (&.any? (bit.complement n/even?) sample)) - (&.empty? (&.filter (bit.complement n/even?) sample))) - (&.any? (bit.complement n/even?) sample))) - - (test "Any element of the list can be considered its member." - (let [elem (maybe.assume (&.nth idx sample))] - (&.member? number.equivalence sample elem))) + (_.test "The size function should correctly portray the size of the list." + (n/= size (/.size sample))) + (_.test "The repeat function should produce as many elements as asked of it." + (n/= size (/.size (/.repeat size [])))) + (_.test "Reversing a list does not change it's size." + (n/= (/.size sample) + (/.size (/.reverse sample)))) + (_.test "Reversing a list twice results in the original list." + (/@= sample + (/.reverse (/.reverse sample)))) + (_.test "Filtering by a predicate and its complement should result in a number of elements equal to the original list." + (and (n/= (/.size sample) + (n/+ (/.size (/.filter n/even? sample)) + (/.size (/.filter (bit.complement n/even?) sample)))) + (let [[plus minus] (/.partition n/even? sample)] + (n/= (/.size sample) + (n/+ (/.size plus) + (/.size minus)))))) + (_.test "If every element in a list satisfies a predicate, there can't be any that satisfy its complement." + (if (/.every? n/even? sample) + (and (not (/.any? (bit.complement n/even?) sample)) + (/.empty? (/.filter (bit.complement n/even?) sample))) + (/.any? (bit.complement n/even?) sample))) + (_.test "Any element of the list can be considered its member." + (let [elem (maybe.assume (/.nth idx sample))] + (/.member? nat.equivalence sample elem))) + (_.test "Appending the head and the tail should yield the original list." + (let [head (maybe.assume (/.head sample)) + tail (maybe.assume (/.tail sample))] + (/@= sample + (#.Cons head tail)))) + (_.test "Appending the inits and the last should yield the original list." + (let [inits (maybe.assume (/.inits sample)) + last (maybe.assume (/.last sample))] + (/@= sample + (/@compose inits (list last))))) + (_.test "Splitting a list into chunks and re-appending them should yield the original list." + (let [[left right] (/.split idx sample) + [left' right'] (/.split-with n/even? sample)] + (and (/@= sample + (/@compose left right)) + (/@= sample + (/@compose left' right')) + (/@= sample + (/@compose (/.take idx sample) + (/.drop idx sample))) + (/@= sample + (/@compose (/.take-while n/even? sample) + (/.drop-while n/even? sample))) + ))) + (_.test "Segmenting the list in pairs should yield as many elements as N/2." + (n/= (n// 2 size) + (/.size (/.as-pairs sample)))) + (_.test "Sorting a list shouldn't change it's size." + (n/= (/.size sample) + (/.size (/.sort n/< sample)))) + (_.test "Sorting a list with one order should yield the reverse of sorting it with the opposite order." + (/@= (/.sort n/< sample) + (/.reverse (/.sort n/> sample)))) + (_.test "If you zip 2 lists, the result's size will be that of the smaller list." + (n/= (/.size (/.zip2 sample other-sample)) + (n/min (/.size sample) (/.size other-sample)))) + (_.test "I can pair-up elements of a list in order." + (let [zipped (/.zip2 sample other-sample) + num-zipper (/.size zipped)] + (and (|> zipped (/@map product.left) (/@= (/.take num-zipper sample))) + (|> zipped (/@map product.right) (/@= (/.take num-zipper other-sample)))))) + (_.test "You can generate indices for any size, and they will be in ascending order." + (let [indices (/.indices size)] + (and (n/= size (/.size indices)) + (/@= indices + (/.sort n/< indices)) + (/.every? (n/= (dec size)) + (/.zip2-with n/+ + indices + (/.sort n/> indices))) + ))) + (_.test "The 'interpose' function places a value between every member of a list." + (let [sample+ (/.interpose separator sample)] + (and (n/= (|> size (n/* 2) dec) + (/.size sample+)) + (|> sample+ /.as-pairs (/@map product.right) (/.every? (n/= separator)))))) + (_.test "You can find any value that satisfies some criterium, if such values exist in the list." + (case (/.find n/even? sample) + (#.Some found) + (and (n/even? found) + (/.any? n/even? sample) + (not (/.every? (bit.complement n/even?) sample))) + + #.None + (and (not (/.any? n/even? sample)) + (/.every? (bit.complement n/even?) sample)))) + (_.test "You can iteratively construct a list, generating values until you're done." + (/@= (/.n/range 0 (dec size)) + (/.iterate (function (_ n) (if (n/< size n) (#.Some (inc n)) #.None)) + 0))) + (_.test "Can enumerate all elements in a list." + (let [enum-sample (/.enumerate sample)] + (and (/@= (/.indices (/.size enum-sample)) + (/@map product.left enum-sample)) + (/@= sample + (/@map product.right enum-sample))))) + (do r.monad + [from (|> r.nat (:: @ map (n/% 10))) + to (|> r.nat (:: @ map (n/% 10)))] + (_.test "Ranges can be constructed forward and backwards." + (and (/@= (/.n/range from to) + (/.reverse (/.n/range to from))) + (let [from (.int from) + to (.int to) + (^open "/@.") (/.equivalence int.equivalence)] + (/@= (/.i/range from to) + (/.reverse (/.i/range to from))))))) )))) - -(context: "Lists: Part 2" - (<| (times 100) - (do @ - [size bounded-size - idx (:: @ map (n/% size) r.nat) - sample (r.list size r.nat) - other-size bounded-size - other-sample (r.list other-size r.nat) - separator r.nat - #let [(^open ".") (&.equivalence number.equivalence) - (^open "&;.") &.functor]] - ($_ seq - (test "Appending the head and the tail should yield the original list." - (let [head (maybe.assume (&.head sample)) - tail (maybe.assume (&.tail sample))] - (= sample - (#.Cons head tail)))) - - (test "Appending the inits and the last should yield the original list." - (let [(^open ".") &.monoid - inits (maybe.assume (&.inits sample)) - last (maybe.assume (&.last sample))] - (= sample - (compose inits (list last))))) - - (test "Functor should go over every element of the list." - (let [(^open ".") &.functor - there (map inc sample) - back-again (map dec there)] - (and (not (= sample there)) - (= sample back-again)))) - - (test "Splitting a list into chunks and re-appending them should yield the original list." - (let [(^open ".") &.monoid - [left right] (&.split idx sample) - [left' right'] (&.split-with n/even? sample)] - (and (= sample - (compose left right)) - (= sample - (compose left' right')) - (= sample - (compose (&.take idx sample) - (&.drop idx sample))) - (= sample - (compose (&.take-while n/even? sample) - (&.drop-while n/even? sample))) - ))) - - (test "Segmenting the list in pairs should yield as many elements as N/2." - (n/= (n// 2 size) - (&.size (&.as-pairs sample)))) - - (test "Sorting a list shouldn't change it's size." - (n/= (&.size sample) - (&.size (&.sort n/< sample)))) - - (test "Sorting a list with one order should yield the reverse of sorting it with the opposite order." - (= (&.sort n/< sample) - (&.reverse (&.sort n/> sample)))) - )))) - -(context: "Lists: Part 3" - (<| (times 100) - (do @ - [size bounded-size - idx (:: @ map (n/% size) r.nat) - sample (r.list size r.nat) - other-size bounded-size - other-sample (r.list other-size r.nat) - separator r.nat - from (|> r.nat (:: @ map (n/% 10))) - to (|> r.nat (:: @ map (n/% 10))) - #let [(^open ".") (&.equivalence number.equivalence) - (^open "&;.") &.functor]] - ($_ seq - (test "If you zip 2 lists, the result's size will be that of the smaller list." - (n/= (&.size (&.zip2 sample other-sample)) - (n/min (&.size sample) (&.size other-sample)))) - - (test "I can pair-up elements of a list in order." - (let [(^open ".") &.functor - zipped (&.zip2 sample other-sample) - num-zipper (&.size zipped)] - (and (|> zipped (map product.left) (= (&.take num-zipper sample))) - (|> zipped (map product.right) (= (&.take num-zipper other-sample)))))) - - (test "You can generate indices for any size, and they will be in ascending order." - (let [(^open ".") &.functor - indices (&.indices size)] - (and (n/= size (&.size indices)) - (= indices - (&.sort n/< indices)) - (&.every? (n/= (dec size)) - (&.zip2-with n/+ - indices - (&.sort n/> indices))) - ))) - - (test "The 'interpose' function places a value between every member of a list." - (let [(^open ".") &.functor - sample+ (&.interpose separator sample)] - (and (n/= (|> size (n/* 2) dec) - (&.size sample+)) - (|> sample+ &.as-pairs (map product.right) (&.every? (n/= separator)))))) - - (test "List append is a monoid." - (let [(^open ".") &.monoid] - (and (= sample (compose identity sample)) - (= sample (compose sample identity)) - (let [[left right] (&.split size (compose sample other-sample))] - (and (= sample left) - (= other-sample right)))))) - - (test "Apply allows you to create singleton lists, and apply lists of functions to lists of values." - (let [(^open ".") &.monad - (^open ".") &.apply] - (and (= (list separator) (wrap separator)) - (= (map inc sample) - (apply (wrap inc) sample))))) - - (test "List concatenation is a monad." - (let [(^open ".") &.monad - (^open ".") &.monoid] - (= (compose sample other-sample) - (join (list sample other-sample))))) - - (test "You can find any value that satisfies some criterium, if such values exist in the list." - (case (&.find n/even? sample) - (#.Some found) - (and (n/even? found) - (&.any? n/even? sample) - (not (&.every? (bit.complement n/even?) sample))) - - #.None - (and (not (&.any? n/even? sample)) - (&.every? (bit.complement n/even?) sample)))) - - (test "You can iteratively construct a list, generating values until you're done." - (= (&.n/range 0 (dec size)) - (&.iterate (function (_ n) (if (n/< size n) (#.Some (inc n)) #.None)) - 0))) - - (test "Can enumerate all elements in a list." - (let [enum-sample (&.enumerate sample)] - (and (= (&.indices (&.size enum-sample)) - (&;map product.left enum-sample)) - (= sample - (&;map product.right enum-sample))))) - - (test "Ranges can be constructed forward and backwards." - (and (let [(^open "list;.") (&.equivalence number.equivalence)] - (list;= (&.n/range from to) - (&.reverse (&.n/range to from)))) - (let [(^open "list;.") (&.equivalence number.equivalence) - from (.int from) - to (.int to)] - (list;= (&.i/range from to) - (&.reverse (&.i/range to from)))))) - )))) - -## TODO: Add again once new-luxc becomes the standard compiler. -(context: "Monad transformer" - (let [lift (&.lift io.monad) - (^open "io;.") io.monad] - (test "Can add list functionality to any monad." - (|> (io.run (do (&.ListT io.monad) - [a (lift (io;wrap +123)) - b (wrap +456)] - (wrap (i/+ a b)))) - (case> (^ (list +579)) #1 - _ #0))))) diff --git a/stdlib/source/test/lux/data/collection/queue.lux b/stdlib/source/test/lux/data/collection/queue.lux index f84246a7f..c377fccc3 100644 --- a/stdlib/source/test/lux/data/collection/queue.lux +++ b/stdlib/source/test/lux/data/collection/queue.lux @@ -1,54 +1,64 @@ (.module: [lux #* + data/text/format + ["_" test (#+ Test)] [control - [monad (#+ do Monad)]] + [monad (#+ do)] + {[0 #test] + [/ + ["$." equivalence] + ["$." functor (#+ Injection)]]}] [data - ["." number] - [collection - ["&" queue]]] + [number + ["." nat]]] [math ["r" random]]] - lux/test) + {1 + ["." /]}) -(context: "Queues" - (<| (times 100) - (do @ +(def: injection + (Injection /.Queue) + (|>> list /.from-list)) + +(def: #export test + Test + (<| (_.context (%name (name-of /.Queue))) + (do r.monad [size (:: @ map (n/% 100) r.nat) sample (r.queue size r.nat) non-member (|> r.nat - (r.filter (|>> (&.member? number.equivalence sample) not)))] - ($_ seq - (test "I can query the size of a queue (and empty queues have size 0)." - (if (n/= 0 size) - (&.empty? sample) - (n/= size (&.size sample)))) - - (test "Enqueueing and dequeing affects the size of queues." - (and (n/= (inc size) (&.size (&.push non-member sample))) - (or (&.empty? sample) - (n/= (dec size) (&.size (&.pop sample)))) - (n/= size (&.size (&.pop (&.push non-member sample)))))) - - (test "Transforming to/from list can't change the queue." - (let [(^open "&;.") (&.equivalence number.equivalence)] - (|> sample - &.to-list &.from-list - (&;= sample)))) - - (test "I can always peek at a non-empty queue." - (case (&.peek sample) - #.None (&.empty? sample) - (#.Some _) #1)) - - (test "I can query whether an element belongs to a queue." - (and (not (&.member? number.equivalence sample non-member)) - (&.member? number.equivalence (&.push non-member sample) - non-member) - (case (&.peek sample) - #.None - (&.empty? sample) - - (#.Some first) - (and (&.member? number.equivalence sample first) - (not (&.member? number.equivalence (&.pop sample) first)))))) + (r.filter (|>> (/.member? nat.equivalence sample) not)))] + ($_ _.and + ($equivalence.spec (/.equivalence nat.equivalence) (r.queue size r.nat)) + ($functor.spec ..injection /.equivalence /.functor) + + (_.test "I can query the size of a queue (and empty queues have size 0)." + (if (n/= 0 size) + (/.empty? sample) + (n/= size (/.size sample)))) + (_.test "Enqueueing and dequeing affects the size of queues." + (and (n/= (inc size) (/.size (/.push non-member sample))) + (or (/.empty? sample) + (n/= (dec size) (/.size (/.pop sample)))) + (n/= size (/.size (/.pop (/.push non-member sample)))))) + (_.test "Transforming to/from list can't change the queue." + (let [(^open "/;.") (/.equivalence nat.equivalence)] + (|> sample + /.to-list /.from-list + (/;= sample)))) + (_.test "I can always peek at a non-empty queue." + (case (/.peek sample) + #.None (/.empty? sample) + (#.Some _) #1)) + (_.test "I can query whether an element belongs to a queue." + (and (not (/.member? nat.equivalence sample non-member)) + (/.member? nat.equivalence (/.push non-member sample) + non-member) + (case (/.peek sample) + #.None + (/.empty? sample) + + (#.Some first) + (and (/.member? nat.equivalence sample first) + (not (/.member? nat.equivalence (/.pop sample) first)))))) )))) diff --git a/stdlib/source/test/lux/data/collection/queue/priority.lux b/stdlib/source/test/lux/data/collection/queue/priority.lux index 3868a01a8..df24b8368 100644 --- a/stdlib/source/test/lux/data/collection/queue/priority.lux +++ b/stdlib/source/test/lux/data/collection/queue/priority.lux @@ -1,57 +1,56 @@ (.module: [lux #* + data/text/format + ["_" test (#+ Test)] [control - ["." monad (#+ do Monad)]] + ["." monad (#+ do)]] [data - [number - ["." nat]] ["." maybe] - [collection - [queue - ["&" priority]]]] + [number + ["." nat]]] [math - ["r" random]]] - lux/test) + ["r" random (#+ Random)]]] + {1 + ["." / (#+ Queue)]}) -(def: (gen-queue size) - (-> Nat (r.Random (&.Queue Nat))) +(def: #export (queue size) + (-> Nat (Random (Queue Nat))) (do r.monad [inputs (r.list size r.nat)] (monad.fold @ (function (_ head tail) (do @ [priority r.nat] - (wrap (&.push priority head tail)))) - &.empty + (wrap (/.push priority head tail)))) + /.empty inputs))) -(context: "Queues" - (<| (times 100) - (do @ +(def: #export test + Test + (<| (_.context (%name (name-of /.Queue))) + (do r.monad [size (|> r.nat (:: @ map (n/% 100))) - sample (gen-queue size) + sample (..queue size) non-member-priority r.nat - non-member (|> r.nat (r.filter (|>> (&.member? nat.equivalence sample) not)))] - ($_ seq - (test "I can query the size of a queue (and empty queues have size 0)." - (n/= size (&.size sample))) - - (test "Enqueueing and dequeing affects the size of queues." - (and (n/= (inc size) - (&.size (&.push non-member-priority non-member sample))) - (or (n/= 0 (&.size sample)) - (n/= (dec size) - (&.size (&.pop sample)))))) - - (test "I can query whether an element belongs to a queue." - (and (and (not (&.member? nat.equivalence sample non-member)) - (&.member? nat.equivalence - (&.push non-member-priority non-member sample) - non-member)) - (or (n/= 0 (&.size sample)) - (and (&.member? nat.equivalence - sample - (maybe.assume (&.peek sample))) - (not (&.member? nat.equivalence - (&.pop sample) - (maybe.assume (&.peek sample)))))))) + non-member (|> r.nat (r.filter (|>> (/.member? nat.equivalence sample) not)))] + ($_ _.and + (_.test "I can query the size of a queue (and empty queues have size 0)." + (n/= size (/.size sample))) + (_.test "Enqueueing and dequeing affects the size of queues." + (and (n/= (inc size) + (/.size (/.push non-member-priority non-member sample))) + (or (n/= 0 (/.size sample)) + (n/= (dec size) + (/.size (/.pop sample)))))) + (_.test "I can query whether an element belongs to a queue." + (and (and (not (/.member? nat.equivalence sample non-member)) + (/.member? nat.equivalence + (/.push non-member-priority non-member sample) + non-member)) + (or (n/= 0 (/.size sample)) + (and (/.member? nat.equivalence + sample + (maybe.assume (/.peek sample))) + (not (/.member? nat.equivalence + (/.pop sample) + (maybe.assume (/.peek sample)))))))) )))) diff --git a/stdlib/source/test/lux/data/collection/row.lux b/stdlib/source/test/lux/data/collection/row.lux index f4c7ad3a0..cf678e0b4 100644 --- a/stdlib/source/test/lux/data/collection/row.lux +++ b/stdlib/source/test/lux/data/collection/row.lux @@ -1,82 +1,76 @@ (.module: [lux #* + data/text/format + ["_" test (#+ Test)] [control - [monad (#+ Monad do)]] + [monad (#+ do)] + {[0 #test] + [/ + ["$." equivalence] + ["$." monoid] + ["$." fold] + ["$." functor (#+ Injection)] + ["$." apply] + ["$." monad]]}] [data - ["." number] ["." maybe] + [number + ["." nat]] [collection - ["&" row] - ["." list ("#;." fold)]]] + ["." list ("#@." fold)]]] [math ["r" random]]] - lux/test) + {1 + ["." / ("#@." monad)]}) -(context: "Rows" - (<| (times 100) - (do @ - [size (|> r.nat (:: @ map (|>> (n/% 100) (n/max 1)))) - idx (|> r.nat (:: @ map (n/% size))) - sample (r.row size r.nat) - other-sample (r.row size r.nat) - non-member (|> r.nat (r.filter (|>> (&.member? number.equivalence sample) not))) - #let [(^open "&;.") (&.equivalence number.equivalence) - (^open "&;.") &.apply - (^open "&;.") &.monad - (^open "&;.") &.fold - (^open "&;.") &.monoid]] - ($_ seq - (test "Can query size of row." - (if (&.empty? sample) - (and (n/= 0 size) - (n/= 0 (&.size sample))) - (n/= size (&.size sample)))) +(def: #export test + Test + (<| (_.context (%name (name-of /.Row))) + (do r.monad + [size (|> r.nat (:: @ map (|>> (n/% 100) (n/max 1))))] + ($_ _.and + ($equivalence.spec (/.equivalence nat.equivalence) (r.row size r.nat)) + ($monoid.spec (/.equivalence nat.equivalence) /.monoid (r.row size r.nat)) + ($fold.spec /@wrap /.equivalence /.fold) + ($functor.spec /@wrap /.equivalence /.functor) + ($apply.spec /@wrap /.equivalence /.apply) + ($monad.spec /@wrap /.equivalence /.monad) - (test "Can add and remove elements to rows." - (and (n/= (inc size) (&.size (&.add non-member sample))) - (n/= (dec size) (&.size (&.pop sample))))) - - (test "Can put and get elements into rows." - (|> sample - (&.put idx non-member) - (&.nth idx) - maybe.assume - (is? non-member))) - - (test "Can update elements of rows." - (|> sample - (&.put idx non-member) (&.update idx inc) - (&.nth idx) maybe.assume - (n/= (inc non-member)))) - - (test "Can safely transform to/from lists." - (|> sample &.to-list &.from-list (&;= sample))) - - (test "Can identify members of a row." - (and (not (&.member? number.equivalence sample non-member)) - (&.member? number.equivalence (&.add non-member sample) non-member))) - - (test "Can fold over elements of row." - (n/= (list;fold n/+ 0 (&.to-list sample)) - (&;fold n/+ 0 sample))) - - (test "Functor goes over every element." - (let [there (&;map inc sample) - back-again (&;map dec there)] - (and (not (&;= sample there)) - (&;= sample back-again)))) - - (test "Apply allows you to create singleton rows, and apply rows of functions to rows of values." - (and (&;= (&.row non-member) (&;wrap non-member)) - (&;= (&;map inc sample) (&;apply (&;wrap inc) sample)))) - - (test "Row concatenation is a monad." - (&;= (&;compose sample other-sample) - (&;join (&.row sample other-sample)))) - - (test "Can reverse." - (and (not (&;= sample - (&.reverse sample))) - (not (&;= sample - (&.reverse (&.reverse sample)))))) + (do @ + [idx (|> r.nat (:: @ map (n/% size))) + sample (r.row size r.nat) + other-sample (r.row size r.nat) + non-member (|> r.nat (r.filter (|>> (/.member? nat.equivalence sample) not))) + #let [(^open "/@.") (/.equivalence nat.equivalence)]] + ($_ _.and + (_.test "Can query size of row." + (if (/.empty? sample) + (and (n/= 0 size) + (n/= 0 (/.size sample))) + (n/= size (/.size sample)))) + (_.test "Can add and remove elements to rows." + (and (n/= (inc size) (/.size (/.add non-member sample))) + (n/= (dec size) (/.size (/.pop sample))))) + (_.test "Can put and get elements into rows." + (|> sample + (/.put idx non-member) + (/.nth idx) + maybe.assume + (is? non-member))) + (_.test "Can update elements of rows." + (|> sample + (/.put idx non-member) (/.update idx inc) + (/.nth idx) maybe.assume + (n/= (inc non-member)))) + (_.test "Can safely transform to/from lists." + (|> sample /.to-list /.from-list (/@= sample))) + (_.test "Can identify members of a row." + (and (not (/.member? nat.equivalence sample non-member)) + (/.member? nat.equivalence (/.add non-member sample) non-member))) + (_.test "Can reverse." + (and (not (/@= sample + (/.reverse sample))) + (not (/@= sample + (/.reverse (/.reverse sample)))))) + )) )))) diff --git a/stdlib/source/test/lux/data/collection/sequence.lux b/stdlib/source/test/lux/data/collection/sequence.lux index 90971d2e9..edacef996 100644 --- a/stdlib/source/test/lux/data/collection/sequence.lux +++ b/stdlib/source/test/lux/data/collection/sequence.lux @@ -1,104 +1,100 @@ (.module: [lux #* + data/text/format + ["_" test (#+ Test)] [control - [monad (#+ do Monad)] - comonad] + comonad + [monad (#+ do Monad)]] [data ["." maybe] [number - ["." nat ("#;." codec)]] - ["." text ("#;." monoid)] + ["." nat ("#@." decimal)]] + ["." text ("#@." monoid)] [collection - ["." list] - ["&" sequence]]] + ["." list]]] [math ["r" random]]] - lux/test) + {1 + ["." /]}) -(context: "Sequences" - (<| (times 100) - (do @ +(def: #export test + Test + (<| (_.context (%name (name-of /.Sequence))) + (do r.monad [size (|> r.nat (:: @ map (|>> (n/% 100) (n/max 2)))) offset (|> r.nat (:: @ map (n/% 100))) factor (|> r.nat (:: @ map (|>> (n/% 100) (n/max 2)))) elem r.nat cycle-seed (r.list size r.nat) cycle-sample-idx (|> r.nat (:: @ map (n/% 1000))) - #let [(^open "List;.") (list.equivalence number.equivalence) - sample0 (&.iterate inc 0) - sample1 (&.iterate inc offset)]] - ($_ seq - (test "Can move along a sequence and take slices off it." - (and (and (List;= (list.n/range 0 (dec size)) - (&.take size sample0)) - (List;= (list.n/range offset (dec (n/+ offset size))) - (&.take size (&.drop offset sample0))) - (let [[drops takes] (&.split size sample0)] - (and (List;= (list.n/range 0 (dec size)) - drops) - (List;= (list.n/range size (dec (n/* 2 size))) - (&.take size takes))))) - (and (List;= (list.n/range 0 (dec size)) - (&.take-while (n/< size) sample0)) - (List;= (list.n/range offset (dec (n/+ offset size))) - (&.take-while (n/< (n/+ offset size)) - (&.drop-while (n/< offset) sample0))) - (let [[drops takes] (&.split-while (n/< size) sample0)] - (and (List;= (list.n/range 0 (dec size)) - drops) - (List;= (list.n/range size (dec (n/* 2 size))) - (&.take-while (n/< (n/* 2 size)) takes))))) - )) - - (test "Can repeat any element and infinite number of times." - (n/= elem (&.nth offset (&.repeat elem)))) - - (test "Can obtain the head & tail of a sequence." - (and (n/= offset (&.head sample1)) - (List;= (list.n/range (inc offset) (n/+ offset size)) - (&.take size (&.tail sample1))))) - - (test "Can filter sequences." - (and (n/= (n/* 2 offset) - (&.nth offset - (&.filter n/even? sample0))) - (let [[evens odds] (&.partition n/even? (&.iterate inc 0))] - (and (n/= (n/* 2 offset) - (&.nth offset evens)) - (n/= (inc (n/* 2 offset)) - (&.nth offset odds)))))) - - (test "Functor goes over 'all' elements in a sequence." - (let [(^open "&;.") &.functor - there (&;map (n/* factor) sample0) - back-again (&;map (n// factor) there)] - (and (not (List;= (&.take size sample0) - (&.take size there))) - (List;= (&.take size sample0) - (&.take size back-again))))) - - (test "CoMonad produces a value for every element in a sequence." - (let [(^open "&;.") &.functor] - (List;= (&.take size (&;map (n/* factor) sample1)) - (&.take size - (be &.comonad - [inputs sample1] - (n/* factor (&.head inputs))))))) - - (test "'unfold' generalizes 'iterate'." - (let [(^open "&;.") &.functor - (^open "List;.") (list.equivalence text.equivalence)] - (List;= (&.take size - (&;map nat;encode (&.iterate inc offset))) - (&.take size - (&.unfold (function (_ n) [(inc n) (nat;encode n)]) - offset))))) - - (test "Can cycle over the same elements as an infinite sequence." - (|> (&.cycle cycle-seed) - maybe.assume - (&.nth cycle-sample-idx) - (n/= (|> cycle-seed - (list.nth (n/% size cycle-sample-idx)) - maybe.assume)))) + #let [(^open "list@.") (list.equivalence nat.equivalence) + sample0 (/.iterate inc 0) + sample1 (/.iterate inc offset)]] + ($_ _.and + (_.test "Can move along a sequence and take slices off it." + (and (and (list@= (list.n/range 0 (dec size)) + (/.take size sample0)) + (list@= (list.n/range offset (dec (n/+ offset size))) + (/.take size (/.drop offset sample0))) + (let [[drops takes] (/.split size sample0)] + (and (list@= (list.n/range 0 (dec size)) + drops) + (list@= (list.n/range size (dec (n/* 2 size))) + (/.take size takes))))) + (and (list@= (list.n/range 0 (dec size)) + (/.take-while (n/< size) sample0)) + (list@= (list.n/range offset (dec (n/+ offset size))) + (/.take-while (n/< (n/+ offset size)) + (/.drop-while (n/< offset) sample0))) + (let [[drops takes] (/.split-while (n/< size) sample0)] + (and (list@= (list.n/range 0 (dec size)) + drops) + (list@= (list.n/range size (dec (n/* 2 size))) + (/.take-while (n/< (n/* 2 size)) takes))))) + )) + (_.test "Can repeat any element and infinite number of times." + (n/= elem (/.nth offset (/.repeat elem)))) + (_.test "Can obtain the head & tail of a sequence." + (and (n/= offset (/.head sample1)) + (list@= (list.n/range (inc offset) (n/+ offset size)) + (/.take size (/.tail sample1))))) + (_.test "Can filter sequences." + (and (n/= (n/* 2 offset) + (/.nth offset + (/.filter n/even? sample0))) + (let [[evens odds] (/.partition n/even? (/.iterate inc 0))] + (and (n/= (n/* 2 offset) + (/.nth offset evens)) + (n/= (inc (n/* 2 offset)) + (/.nth offset odds)))))) + (_.test "Functor goes over 'all' elements in a sequence." + (let [(^open "/@.") /.functor + there (/@map (n/* factor) sample0) + back-again (/@map (n// factor) there)] + (and (not (list@= (/.take size sample0) + (/.take size there))) + (list@= (/.take size sample0) + (/.take size back-again))))) + (_.test "CoMonad produces a value for every element in a sequence." + (let [(^open "/@.") /.functor] + (list@= (/.take size (/@map (n/* factor) sample1)) + (/.take size + (be /.comonad + [inputs sample1] + (n/* factor (/.head inputs))))))) + (_.test "'unfold' generalizes 'iterate'." + (let [(^open "/@.") /.functor + (^open "list@.") (list.equivalence text.equivalence)] + (list@= (/.take size + (/@map nat@encode (/.iterate inc offset))) + (/.take size + (/.unfold (function (_ n) [(inc n) (nat@encode n)]) + offset))))) + (_.test "Can cycle over the same elements as an infinite sequence." + (|> (/.cycle cycle-seed) + maybe.assume + (/.nth cycle-sample-idx) + (n/= (|> cycle-seed + (list.nth (n/% size cycle-sample-idx)) + maybe.assume)))) )))) diff --git a/stdlib/source/test/lux/data/collection/set.lux b/stdlib/source/test/lux/data/collection/set.lux index b383f32c2..f319af295 100644 --- a/stdlib/source/test/lux/data/collection/set.lux +++ b/stdlib/source/test/lux/data/collection/set.lux @@ -1,67 +1,74 @@ (.module: [lux #* + data/text/format + ["_" test (#+ Test)] [control - [monad (#+ do Monad)]] + [monad (#+ do)] + {[0 #test] + [/ + ["$." equivalence] + ["$." monoid]]}] [data - ["." number] + [number + ["." nat]] [collection - ["&" set (#+ Set)] ["." list]]] [math ["r" random]]] - lux/test) + {1 + ["." /]}) (def: gen-nat (r.Random Nat) (|> r.nat (:: r.monad map (n/% 100)))) -(context: "Sets" - (<| (times 100) - (do @ - [sizeL gen-nat - sizeR gen-nat - setL (r.set number.hash sizeL gen-nat) - setR (r.set number.hash sizeR gen-nat) - non-member (|> gen-nat - (r.filter (|>> (&.member? setL) not))) - #let [(^open "&;.") &.equivalence]] - ($_ seq - (test "I can query the size of a set." - (and (n/= sizeL (&.size setL)) - (n/= sizeR (&.size setR)))) +(def: #export test + Test + (<| (_.context (%name (name-of /.Set))) + (do r.monad + [size gen-nat] + ($_ _.and + ($equivalence.spec /.equivalence (r.set nat.hash size r.nat)) + ($monoid.spec /.equivalence (/.monoid nat.hash) (r.set nat.hash size r.nat)) - (test "Converting sets to/from lists can't change their values." - (|> setL - &.to-list (&.from-list number.hash) - (&;= setL))) - - (test "Every set is a sub-set of the union of itself with another." - (let [setLR (&.union setL setR)] - (and (&.sub? setLR setL) - (&.sub? setLR setR)))) - - (test "Every set is a super-set of the intersection of itself with another." - (let [setLR (&.intersection setL setR)] - (and (&.super? setLR setL) - (&.super? setLR setR)))) - - (test "Union with the empty set leaves a set unchanged." - (&;= setL - (&.union (&.new number.hash) - setL))) - - (test "Intersection with the empty set results in the empty set." - (let [empty-set (&.new number.hash)] - (&;= empty-set - (&.intersection empty-set setL)))) - - (test "After substracting a set A from another B, no member of A can be a member of B." - (let [sub (&.difference setR setL)] - (not (list.any? (&.member? sub) (&.to-list setR))))) - - (test "Every member of a set must be identifiable." - (and (not (&.member? setL non-member)) - (&.member? (&.add non-member setL) non-member) - (not (&.member? (&.remove non-member (&.add non-member setL)) non-member)))) - )))) + (do r.monad + [sizeL gen-nat + sizeR gen-nat + setL (r.set nat.hash sizeL gen-nat) + setR (r.set nat.hash sizeR gen-nat) + non-member (|> gen-nat + (r.filter (|>> (/.member? setL) not))) + #let [(^open "/@.") /.equivalence]] + ($_ _.and + (_.test "I can query the size of a set." + (and (n/= sizeL (/.size setL)) + (n/= sizeR (/.size setR)))) + (_.test "Converting sets to/from lists can't change their values." + (|> setL + /.to-list (/.from-list nat.hash) + (/@= setL))) + (_.test "Every set is a sub-set of the union of itself with another." + (let [setLR (/.union setL setR)] + (and (/.sub? setLR setL) + (/.sub? setLR setR)))) + (_.test "Every set is a super-set of the intersection of itself with another." + (let [setLR (/.intersection setL setR)] + (and (/.super? setLR setL) + (/.super? setLR setR)))) + (_.test "Union with the empty set leaves a set unchanged." + (/@= setL + (/.union (/.new nat.hash) + setL))) + (_.test "Intersection with the empty set results in the empty set." + (let [empty-set (/.new nat.hash)] + (/@= empty-set + (/.intersection empty-set setL)))) + (_.test "After substracting a set A from another B, no member of A can be a member of B." + (let [sub (/.difference setR setL)] + (not (list.any? (/.member? sub) (/.to-list setR))))) + (_.test "Every member of a set must be identifiable." + (and (not (/.member? setL non-member)) + (/.member? (/.add non-member setL) non-member) + (not (/.member? (/.remove non-member (/.add non-member setL)) non-member)))) + )))))) diff --git a/stdlib/source/test/lux/data/collection/set/ordered.lux b/stdlib/source/test/lux/data/collection/set/ordered.lux index 78d096cef..7f143a9cd 100644 --- a/stdlib/source/test/lux/data/collection/set/ordered.lux +++ b/stdlib/source/test/lux/data/collection/set/ordered.lux @@ -1,98 +1,113 @@ (.module: [lux #* + data/text/format + ["_" test (#+ Test)] [control - [monad (#+ do Monad)]] + [monad (#+ do)] + [order (#+ Order)] + {[0 #test] + [/ + ["$." equivalence]]}] [data - ["." number] - [text - format] + [number + ["." nat]] [collection - ["." set - ["&" ordered]] ["." list]]] [math - ["r" random]]] - lux/test) + ["r" random (#+ Random) ("#@." monad)]]] + {1 + ["." / (#+ Set) + ["." //]]}) (def: gen-nat (r.Random Nat) (|> r.nat (:: r.monad map (n/% 100)))) -(context: "Sets" - (<| (times 100) - (do @ - [sizeL gen-nat - sizeR gen-nat - listL (|> (r.set number.hash sizeL gen-nat) (:: @ map set.to-list)) - listR (|> (r.set number.hash sizeR gen-nat) (:: @ map set.to-list)) - #let [(^open "&;.") &.equivalence - setL (&.from-list number.order listL) - setR (&.from-list number.order listR) - sortedL (list.sort n/< listL) - minL (list.head sortedL) - maxL (list.last sortedL)]] - ($_ seq - (test "I can query the size of a set." - (n/= sizeL (&.size setL))) - - (test "Can query minimum value." - (case [(&.min setL) minL] - [#.None #.None] - #1 - - [(#.Some reference) (#.Some sample)] - (n/= reference sample) - - _ - #0)) - - (test "Can query maximum value." - (case [(&.max setL) maxL] - [#.None #.None] - #1 - - [(#.Some reference) (#.Some sample)] - (n/= reference sample) - - _ - #0)) - - (test "Converting sets to/from lists can't change their values." - (|> setL - &.to-list (&.from-list number.order) - (&;= setL))) - - (test "Order is preserved." - (let [listL (&.to-list setL) - (^open "L/.") (list.equivalence number.equivalence)] - (L/= listL - (list.sort n/< listL)))) - - (test "Every set is a sub-set of the union of itself with another." - (let [setLR (&.union setL setR)] - (and (&.sub? setLR setL) - (&.sub? setLR setR)))) - - (test "Every set is a super-set of the intersection of itself with another." - (let [setLR (&.intersection setL setR)] - (and (&.super? setLR setL) - (&.super? setLR setR)))) - - (test "Union with the empty set leaves a set unchanged." - (&;= setL - (&.union (&.new number.order) - setL))) - - (test "Intersection with the empty set results in the empty set." - (let [empty-set (&.new number.order)] - (&;= empty-set - (&.intersection empty-set setL)))) - - (test "After substracting a set A from another B, no member of A can be a member of B." - (let [sub (&.difference setR setL)] - (not (list.any? (&.member? sub) (&.to-list setR))))) - - (test "Every member of a set must be identifiable." - (list.every? (&.member? setL) (&.to-list setL))) - )))) +(def: #export (set &order gen-value size) + (All [a] (-> (Order a) (Random a) Nat (Random (Set a)))) + (case size + 0 + (r@wrap (/.new &order)) + + _ + (do r.monad + [partial (set &order gen-value (dec size)) + value (r.filter (|>> (/.member? partial) not) + gen-value)] + (wrap (/.add value partial))))) + +(def: #export test + Test + (<| (_.context (%name (name-of /.Set))) + ($_ _.and + (do r.monad + [size gen-nat] + ($_ _.and + ($equivalence.spec /.equivalence (..set nat.order r.nat size)) + )) + (do r.monad + [sizeL gen-nat + sizeR gen-nat + listL (|> (r.set nat.hash sizeL gen-nat) (:: @ map //.to-list)) + listR (|> (r.set nat.hash sizeR gen-nat) (:: @ map //.to-list)) + #let [(^open "/@.") /.equivalence + setL (/.from-list nat.order listL) + setR (/.from-list nat.order listR) + sortedL (list.sort n/< listL) + minL (list.head sortedL) + maxL (list.last sortedL)]] + ($_ _.and + (_.test "I can query the size of a set." + (n/= sizeL (/.size setL))) + (_.test "Can query minimum value." + (case [(/.min setL) minL] + [#.None #.None] + true + + [(#.Some reference) (#.Some sample)] + (n/= reference sample) + + _ + false)) + (_.test "Can query maximum value." + (case [(/.max setL) maxL] + [#.None #.None] + true + + [(#.Some reference) (#.Some sample)] + (n/= reference sample) + + _ + false)) + (_.test "Converting sets to/from lists can't change their values." + (|> setL + /.to-list (/.from-list nat.order) + (/@= setL))) + (_.test "Order is preserved." + (let [listL (/.to-list setL) + (^open "list@.") (list.equivalence nat.equivalence)] + (list@= listL + (list.sort n/< listL)))) + (_.test "Every set is a sub-set of the union of itself with another." + (let [setLR (/.union setL setR)] + (and (/.sub? setLR setL) + (/.sub? setLR setR)))) + (_.test "Every set is a super-set of the intersection of itself with another." + (let [setLR (/.intersection setL setR)] + (and (/.super? setLR setL) + (/.super? setLR setR)))) + (_.test "Union with the empty set leaves a set unchanged." + (/@= setL + (/.union (/.new nat.order) + setL))) + (_.test "Intersection with the empty set results in the empty set." + (let [empty-set (/.new nat.order)] + (/@= empty-set + (/.intersection empty-set setL)))) + (_.test "After substracting a set A from another B, no member of A can be a member of B." + (let [sub (/.difference setR setL)] + (not (list.any? (/.member? sub) (/.to-list setR))))) + (_.test "Every member of a set must be identifiable." + (list.every? (/.member? setL) (/.to-list setL))) + ))))) diff --git a/stdlib/source/test/lux/data/collection/stack.lux b/stdlib/source/test/lux/data/collection/stack.lux index d203b4246..0a6fcf698 100644 --- a/stdlib/source/test/lux/data/collection/stack.lux +++ b/stdlib/source/test/lux/data/collection/stack.lux @@ -1,46 +1,69 @@ (.module: [lux #* + data/text/format + ["_" test (#+ Test)] [control - [monad (#+ do)]] + [monad (#+ do)] + {[0 #test] + [/ + ["$." equivalence] + ["$." functor (#+ Injection)]]}] [data ["." maybe] - [collection - ["&" stack]]] + [number + ["." nat]]] [math ["r" random]]] - lux/test) + {1 + ["." /]}) + +(def: (injection value) + (Injection /.Stack) + (/.push value /.empty)) (def: gen-nat (r.Random Nat) (|> r.nat (:: r.monad map (n/% 100)))) -(context: "Stacks" - (<| (times 100) - (do @ +(def: #export test + Test + (<| (_.context (%name (name-of /.Stack))) + (do r.monad [size gen-nat sample (r.stack size gen-nat) new-top gen-nat] - ($_ seq - (test "Can query the size of a stack." - (n/= size (&.size sample))) - - (test "Can peek inside non-empty stacks." - (case (&.peek sample) - #.None (&.empty? sample) - (#.Some _) (not (&.empty? sample)))) - - (test "Popping empty stacks doesn't change anything. - But, if they're non-empty, the top of the stack is removed." - (let [sample' (&.pop sample)] - (or (n/= (&.size sample) (inc (&.size sample'))) - (and (&.empty? sample) (&.empty? sample'))) - )) - - (test "Pushing onto a stack always increases it by 1, adding a new value at the top." - (and (is? sample - (&.pop (&.push new-top sample))) - (n/= (inc (&.size sample)) (&.size (&.push new-top sample))) - (|> (&.push new-top sample) &.peek maybe.assume - (is? new-top)))) + ($_ _.and + ($equivalence.spec (/.equivalence nat.equivalence) (r.stack size r.nat)) + ($functor.spec ..injection /.equivalence /.functor) + + (_.test "Can query the size of a stack." + (n/= size (/.size sample))) + (_.test "Can peek inside non-empty stacks." + (case (/.peek sample) + #.None (/.empty? sample) + (#.Some _) (not (/.empty? sample)))) + (_.test (format "Popping empty stacks doesn't change anything." + "But, if they're non-empty, the top of the stack is removed.") + (case (/.size sample) + 0 (case (/.pop sample) + #.None + (/.empty? sample) + + (#.Some _) + false) + expected (case (/.pop sample) + (#.Some sample') + (and (n/= expected (/.size sample')) + (not (/.empty? sample))) + + #.None + false))) + (_.test "Pushing onto a stack always increases it by 1, adding a new value at the top." + (and (is? sample + (|> sample (/.push new-top) /.pop maybe.assume)) + (n/= (inc (/.size sample)) + (/.size (/.push new-top sample))) + (|> (/.push new-top sample) /.peek maybe.assume + (is? new-top)))) )))) diff --git a/stdlib/source/test/lux/data/collection/tree/rose.lux b/stdlib/source/test/lux/data/collection/tree/rose.lux index f4ddee14e..383e250b5 100644 --- a/stdlib/source/test/lux/data/collection/tree/rose.lux +++ b/stdlib/source/test/lux/data/collection/tree/rose.lux @@ -1,51 +1,55 @@ (.module: [lux #* + data/text/format + ["_" test (#+ Test)] [control - [monad (#+ do Monad)]] + [monad (#+ do)] + {[0 #test] + [/ + ["$." equivalence] + ["$." fold] + ["$." functor]]}] [data - ["." product] - ["." number] - ["." text ("#;." equivalence) - format] + [number + ["." nat]] [collection - ["." list ("#;." functor fold)] - [tree - ["&" rose]]]] + ["." list ("#@." functor fold)]]] [math - ["r" random]]] - lux/test) + ["r" random (#+ Random)]]] + {1 + ["." / (#+ Tree)]}) -(def: gen-tree - (r.Random [Nat (&.Tree Nat)]) - (r.rec - (function (_ gen-tree) - (r.either (:: r.monad map (|>> &.leaf [1]) r.nat) - (do r.monad - [value r.nat - num-children (|> r.nat (:: @ map (n/% 3))) - children' (r.list num-children gen-tree) - #let [size' (list;fold n/+ 0 (list;map product.left children')) - children (list;map product.right children')]] - (wrap [(inc size') - (&.branch value children)])) - )))) +(def: #export (tree size gen-value) + (All [a] (-> Nat (Random a) (Random (Tree a)))) + (let [singleton (:: r.monad map /.leaf gen-value)] + (case size + 0 + singleton + + 1 + singleton + + _ + (do r.monad + [value gen-value + children (r.list (n/+ 2 (n/% 2 size)) + (tree (n// 2 size) gen-value))] + (wrap (/.branch value children))) + ))) -(context: "Trees" - (<| (times 100) - (do @ - [[size sample] gen-tree - #let [(^open "&;.") (&.equivalence number.equivalence) - (^open "&;.") &.fold - concat (function (_ addition partial) (format partial (%n addition)))]] - ($_ seq - (test "Can compare trees for equivalence." - (&;= sample sample)) - - (test "Can flatten a tree to get all the nodes as a flat tree." - (n/= size - (list.size (&.flatten sample)))) - - (test "Can fold trees." - (text;= (&;fold concat "" sample) - (list;fold concat "" (&.flatten sample)))) +(def: #export test + Test + (<| (_.context (%name (name-of /.Tree))) + (do r.monad + [size (:: @ map (|>> (n/% 100) (n/max 10)) r.nat)] + ($_ _.and + ($equivalence.spec (/.equivalence nat.equivalence) (..tree size r.nat)) + ($fold.spec /.leaf /.equivalence /.fold) + ($functor.spec /.leaf /.equivalence /.functor) + + (do @ + [sample (..tree size r.nat)] + (_.test "Can flatten a tree to get all the nodes as a flat tree." + (n/= size + (list.size (/.flatten sample))))) )))) diff --git a/stdlib/source/test/lux/data/collection/tree/rose/zipper.lux b/stdlib/source/test/lux/data/collection/tree/rose/zipper.lux index 769e11293..379b17c16 100644 --- a/stdlib/source/test/lux/data/collection/tree/rose/zipper.lux +++ b/stdlib/source/test/lux/data/collection/tree/rose/zipper.lux @@ -1,128 +1,117 @@ (.module: [lux #* + data/text/format + ["_" test (#+ Test)] [control [monad (#+ do Monad)] pipe] [data - ["." number] ["." maybe] - ["." text - format] + ["." text] + [number + ["." nat]] [collection ["." list] [tree - ["." rose - ["&" zipper]]]]] + ["." rose]]]] [math ["r" random]]] - lux/test) - -(def: gen-tree - (r.Random (rose.Tree Nat)) - (r.rec (function (_ gen-tree) - (do r.monad - ## Each branch can have, at most, 1 child. - [size (|> r.nat (:: @ map (n/% 2)))] - (r.and r.nat - (r.list size gen-tree)))))) + ["." //] + {1 + ["." / (#+ Zipper)]} + ) (def: (to-end zipper) - (All [a] (-> (&.Zipper a) (&.Zipper a))) + (All [a] (-> (Zipper a) (Zipper a))) (loop [zipper zipper] - (if (&.end? zipper) + (if (/.end? zipper) zipper - (recur (&.next zipper))))) + (recur (/.next zipper))))) -(context: "Zippers." - (<| (times 100) - (do @ - [sample gen-tree +(def: #export test + Test + (<| (_.context (%name (name-of /.Zipper))) + (do r.monad + [size (:: @ map (|>> (n/% 100) (n/max 10)) r.nat) + sample (//.tree size r.nat) new-val r.nat pre-val r.nat post-val r.nat - #let [(^open "tree/.") (rose.equivalence number.equivalence) - (^open "list;.") (list.equivalence number.equivalence)]] - ($_ seq - (test "Trees can be converted to/from zippers." - (|> sample - &.zip &.unzip - (tree/= sample))) - - (test "Creating a zipper gives you a root node." - (|> sample &.zip &.root?)) - - (test "Can move down inside branches. Can move up from lower nodes." - (let [zipper (&.zip sample)] - (if (&.branch? zipper) - (let [child (|> zipper &.down)] - (and (not (tree/= sample (&.unzip child))) - (|> child &.up (is? zipper) not) - (|> child &.root (is? zipper) not))) - (and (&.leaf? zipper) - (|> zipper (&.prepend-child new-val) &.branch?))))) - - (test "Can prepend and append children." - (let [zipper (&.zip sample)] - (if (&.branch? zipper) - (let [mid-val (|> zipper &.down &.value) - zipper (|> zipper - (&.prepend-child pre-val) - (&.append-child post-val))] - (and (|> zipper &.down &.value (is? pre-val)) - (|> zipper &.down &.right &.value (is? mid-val)) - (|> zipper &.down &.right &.right &.value (is? post-val)) - (|> zipper &.down &.rightmost &.leftmost &.value (is? pre-val)) - (|> zipper &.down &.right &.left &.value (is? pre-val)) - (|> zipper &.down &.rightmost &.value (is? post-val)))) - #1))) - - (test "Can insert children around a node (unless it's root)." - (let [zipper (&.zip sample)] - (if (&.branch? zipper) - (let [mid-val (|> zipper &.down &.value) - zipper (|> zipper - &.down - (&.insert-left pre-val) - maybe.assume - (&.insert-right post-val) - maybe.assume - &.up)] - (and (|> zipper &.down &.value (is? pre-val)) - (|> zipper &.down &.right &.value (is? mid-val)) - (|> zipper &.down &.right &.right &.value (is? post-val)) - (|> zipper &.down &.rightmost &.leftmost &.value (is? pre-val)) - (|> zipper &.down &.right &.left &.value (is? pre-val)) - (|> zipper &.down &.rightmost &.value (is? post-val)))) - (and (|> zipper (&.insert-left pre-val) (case> (#.Some _) #0 - #.None #1)) - (|> zipper (&.insert-right post-val) (case> (#.Some _) #0 - #.None #1)))))) - - (test "Can set and update the value of a node." - (|> sample &.zip (&.set new-val) &.value (n/= new-val))) - - (test "Zipper traversal follows the outline of the tree depth-first." - (list;= (rose.flatten sample) - (loop [zipper (&.zip sample)] - (if (&.end? zipper) - (list (&.value zipper)) - (#.Cons (&.value zipper) - (recur (&.next zipper))))))) - - (test "Backwards zipper traversal yield reverse tree flatten." - (list;= (list.reverse (rose.flatten sample)) - (loop [zipper (to-end (&.zip sample))] - (if (&.root? zipper) - (list (&.value zipper)) - (#.Cons (&.value zipper) - (recur (&.prev zipper))))))) - - (test "Can remove nodes (except root nodes)." - (let [zipper (&.zip sample)] - (if (&.branch? zipper) - (and (|> zipper &.down &.root? not) - (|> zipper &.down &.remove (case> #.None #0 - (#.Some node) (&.root? node)))) - (|> zipper &.remove (case> #.None #1 - (#.Some _) #0))))) + #let [(^open "tree@.") (rose.equivalence nat.equivalence) + (^open "list@.") (list.equivalence nat.equivalence)]] + ($_ _.and + (_.test "Trees can be converted to/from zippers." + (|> sample + /.zip /.unzip + (tree@= sample))) + (_.test "Creating a zipper gives you a root node." + (|> sample /.zip /.root?)) + (_.test "Can move down inside branches. Can move up from lower nodes." + (let [zipper (/.zip sample)] + (if (/.branch? zipper) + (let [child (|> zipper /.down)] + (and (not (tree@= sample (/.unzip child))) + (|> child /.up (is? zipper) not) + (|> child /.root (is? zipper) not))) + (and (/.leaf? zipper) + (|> zipper (/.prepend-child new-val) /.branch?))))) + (_.test "Can prepend and append children." + (let [zipper (/.zip sample)] + (if (/.branch? zipper) + (let [mid-val (|> zipper /.down /.value) + zipper (|> zipper + (/.prepend-child pre-val) + (/.append-child post-val))] + (and (|> zipper /.down /.value (is? pre-val)) + (|> zipper /.down /.right /.value (is? mid-val)) + (|> zipper /.down /.right /.right /.value (is? post-val)) + (|> zipper /.down /.rightmost /.leftmost /.value (is? pre-val)) + (|> zipper /.down /.right /.left /.value (is? pre-val)) + (|> zipper /.down /.rightmost /.value (is? post-val)))) + true))) + (_.test "Can insert children around a node (unless it's root)." + (let [zipper (/.zip sample)] + (if (/.branch? zipper) + (let [mid-val (|> zipper /.down /.value) + zipper (|> zipper + /.down + (/.insert-left pre-val) + maybe.assume + (/.insert-right post-val) + maybe.assume + /.up)] + (and (|> zipper /.down /.value (is? pre-val)) + (|> zipper /.down /.right /.value (is? mid-val)) + (|> zipper /.down /.right /.right /.value (is? post-val)) + (|> zipper /.down /.rightmost /.leftmost /.value (is? pre-val)) + (|> zipper /.down /.right /.left /.value (is? pre-val)) + (|> zipper /.down /.rightmost /.value (is? post-val)))) + (and (|> zipper (/.insert-left pre-val) (case> (#.Some _) false + #.None true)) + (|> zipper (/.insert-right post-val) (case> (#.Some _) false + #.None true)))))) + (_.test "Can set and update the value of a node." + (|> sample /.zip (/.set new-val) /.value (n/= new-val))) + (_.test "Zipper traversal follows the outline of the tree depth-first." + (list@= (rose.flatten sample) + (loop [zipper (/.zip sample)] + (if (/.end? zipper) + (list (/.value zipper)) + (#.Cons (/.value zipper) + (recur (/.next zipper))))))) + (_.test "Backwards zipper traversal yield reverse tree flatten." + (list@= (list.reverse (rose.flatten sample)) + (loop [zipper (to-end (/.zip sample))] + (if (/.root? zipper) + (list (/.value zipper)) + (#.Cons (/.value zipper) + (recur (/.prev zipper))))))) + (_.test "Can remove nodes (except root nodes)." + (let [zipper (/.zip sample)] + (if (/.branch? zipper) + (and (|> zipper /.down /.root? not) + (|> zipper /.down /.remove (case> #.None false + (#.Some node) (/.root? node)))) + (|> zipper /.remove (case> #.None true + (#.Some _) false))))) )))) diff --git a/stdlib/source/test/lux/data/maybe.lux b/stdlib/source/test/lux/data/maybe.lux index 9b3a77ff9..4aa89f85f 100644 --- a/stdlib/source/test/lux/data/maybe.lux +++ b/stdlib/source/test/lux/data/maybe.lux @@ -6,10 +6,10 @@ [monad (#+ do)] {[0 #test] [/ - ["$." functor (#+ Injection Comparison)] + ["$." equivalence] + ["$." functor] ["$." apply] - ["$." monad] - ["$." equivalence]]}] + ["$." monad]]}] [data ["." text format] @@ -19,16 +19,7 @@ [math ["r" random (#+ Random)]]] {1 - ["." / ("#@." monoid)]}) - -(def: injection - (Injection Maybe) - (|>> #.Some)) - -(def: comparison - (Comparison Maybe) - (function (_ ==) - (:: (/.equivalence ==) =))) + ["." / ("#@." monoid monad)]}) (def: #export maybe (All [a] (-> (Random a) (Random (Maybe a)))) @@ -39,9 +30,9 @@ (<| (_.context (%name (name-of .Maybe))) ($_ _.and ($equivalence.spec (/.equivalence nat.equivalence) (..maybe r.nat)) - ($functor.spec ..injection ..comparison /.functor) - ($apply.spec ..injection ..comparison /.apply) - ($monad.spec ..injection ..comparison /.monad) + ($functor.spec /@wrap /.equivalence /.functor) + ($apply.spec /@wrap /.equivalence /.apply) + ($monad.spec /@wrap /.equivalence /.monad) (do r.monad [left r.nat |