aboutsummaryrefslogtreecommitdiff
path: root/stdlib/test
diff options
context:
space:
mode:
Diffstat (limited to 'stdlib/test')
-rw-r--r--stdlib/test/test/lux.lux164
-rw-r--r--stdlib/test/test/lux/cli.lux84
-rw-r--r--stdlib/test/test/lux/codata/env.lux23
-rw-r--r--stdlib/test/test/lux/codata/io.lux21
-rw-r--r--stdlib/test/test/lux/codata/state.lux34
-rw-r--r--stdlib/test/test/lux/codata/struct/stream.lux68
-rw-r--r--stdlib/test/test/lux/concurrency/actor.lux70
-rw-r--r--stdlib/test/test/lux/concurrency/frp.lux54
-rw-r--r--stdlib/test/test/lux/concurrency/promise.lux31
-rw-r--r--stdlib/test/test/lux/concurrency/stm.lux57
-rw-r--r--stdlib/test/test/lux/data/bit.lux65
-rw-r--r--stdlib/test/test/lux/data/bool.lux38
-rw-r--r--stdlib/test/test/lux/data/char.lux47
-rw-r--r--stdlib/test/test/lux/data/error.lux42
-rw-r--r--stdlib/test/test/lux/data/format/json.lux314
-rw-r--r--stdlib/test/test/lux/data/ident.lux53
-rw-r--r--stdlib/test/test/lux/data/identity.lux36
-rw-r--r--stdlib/test/test/lux/data/log.lux32
-rw-r--r--stdlib/test/test/lux/data/maybe.lux49
-rw-r--r--stdlib/test/test/lux/data/number.lux135
-rw-r--r--stdlib/test/test/lux/data/product.lux20
-rw-r--r--stdlib/test/test/lux/data/struct/array.lux130
-rw-r--r--stdlib/test/test/lux/data/struct/dict.lux136
-rw-r--r--stdlib/test/test/lux/data/struct/list.lux191
-rw-r--r--stdlib/test/test/lux/data/struct/queue.lux54
-rw-r--r--stdlib/test/test/lux/data/struct/set.lux67
-rw-r--r--stdlib/test/test/lux/data/struct/stack.lux47
-rw-r--r--stdlib/test/test/lux/data/struct/tree.lux39
-rw-r--r--stdlib/test/test/lux/data/struct/vector.lux84
-rw-r--r--stdlib/test/test/lux/data/struct/zipper.lux127
-rw-r--r--stdlib/test/test/lux/data/sum.lux32
-rw-r--r--stdlib/test/test/lux/data/text.lux150
-rw-r--r--stdlib/test/test/lux/data/text/format.lux22
-rw-r--r--stdlib/test/test/lux/host.lux54
-rw-r--r--stdlib/test/test/lux/lexer.lux133
-rw-r--r--stdlib/test/test/lux/macro/ast.lux31
-rw-r--r--stdlib/test/test/lux/macro/syntax.lux176
-rw-r--r--stdlib/test/test/lux/math.lux45
-rw-r--r--stdlib/test/test/lux/pipe.lux47
-rw-r--r--stdlib/test/test/lux/regex.lux200
-rw-r--r--stdlib/test/test/lux/type.lux41
-rw-r--r--stdlib/test/tests.lux84
42 files changed, 3327 insertions, 0 deletions
diff --git a/stdlib/test/test/lux.lux b/stdlib/test/test/lux.lux
new file mode 100644
index 000000000..947ec5b6f
--- /dev/null
+++ b/stdlib/test/test/lux.lux
@@ -0,0 +1,164 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ lux/test
+ (lux (control monad)
+ (codata [io])
+ [math]
+ (math ["R" random])
+ (data text/format)
+ [compiler]
+ (macro ["s" syntax #+ syntax:])))
+
+(test: "Every value is identical to itself, and the 'id' function doesn't change values in any way."
+ [value R;int]
+ (assert "" (and (== value value)
+ (== value (id value)))))
+
+(test: "Values created separately can't be identical."
+ [x R;int
+ y R;int]
+ (match false (== x y)))
+
+(do-template [category rand-gen inc dec even? odd? = < >]
+ [(test: (format "[" category "] " "Moving up-down or down-up should result in same value.")
+ [value rand-gen]
+ (assert "" (and (|> value inc dec (= value))
+ (|> value dec inc (= value)))))
+
+ (test: (format "[" category "] " "(x+1) > x && (x-1) < x")
+ [value rand-gen]
+ (assert "" (and (|> value inc (> value))
+ (|> value dec (< value)))))
+
+ (test: (format "[" category "] " "Every odd/even number is surrounded by two of the other kind.")
+ [value rand-gen]
+ (assert ""
+ (if (even? value)
+ (and (|> value inc odd?)
+ (|> value dec odd?))
+ (and (|> value inc even?)
+ (|> value dec even?)))))]
+
+ ["Nat" R;nat inc+ dec+ even?+ odd?+ =+ <+ >+]
+ ["Int" R;int inc dec even? odd? = < >]
+ )
+
+(do-template [category rand-gen = < > <= >= min max]
+ [(test: (format "[" category "] " "The symmetry of numerical comparisons.")
+ [x rand-gen
+ y rand-gen]
+ (assert ""
+ (or (= x y)
+ (if (< y x)
+ (> x y)
+ (< x y)))))
+
+ (test: (format "[" category "] " "Minimums and maximums.")
+ [x rand-gen
+ y rand-gen]
+ (assert ""
+ (and (and (<= x (min x y))
+ (<= y (min x y)))
+ (and (>= x (max x y))
+ (>= y (max x y)))
+ )))]
+
+ ["Int" R;int = < > <= >= min max]
+ ["Nat" R;nat =+ <+ >+ <=+ >=+ min+ max+]
+ ["Real" R;real =. <. >. <=. >=. min. max.]
+ ["Frac" R;frac =.. <.. >.. <=.. >=.. min.. max..]
+ )
+
+(do-template [category rand-gen = + - * / <%> > <0> <1> <10> %x <cap> <prep>]
+ [(test: (format "[" category "] " "Additive identity")
+ [x rand-gen]
+ (assert ""
+ (and (|> x (+ <0>) (= x))
+ (|> x (- <0>) (= x)))))
+
+ (test: (format "[" category "] " "Addition & Substraction")
+ [x (:: @ map <prep> rand-gen)
+ y (:: @ map <prep> rand-gen)
+ #let [x (* <10> x)
+ y (* <10> y)
+ cond (and (|> x (- y) (+ y) (= x))
+ (|> x (+ y) (- y) (= x)))
+ _ (if cond
+ []
+ (exec
+ (log! "+- SAMPLE")
+ (log! (format (%x x) " -+ " (%x y) " = " (%x (|> x (- y) (+ y)))))
+ (log! (format (%x x) " +- " (%x y) " = " (%x (|> x (+ y) (- y)))))))
+ ]]
+ (assert ""
+ (and (|> x (- y) (+ y) (= x))
+ (|> x (+ y) (- y) (= x)))))
+
+ (test: (format "[" category "] " "Multiplicative identity")
+ [x rand-gen]
+ (assert ""
+ (and (|> x (* <1>) (= x))
+ (|> x (/ <1>) (= x)))))
+
+ (test: (format "[" category "] " "Multiplication & Division")
+ [x (:: @ map <cap> rand-gen)
+ y (|> rand-gen
+ (:: @ map <cap>)
+ (R;filter (|>. (= <0>) not)))
+ #let [r (<%> y x)
+ x' (- r x)]]
+ (assert ""
+ (or (> x' y)
+ (|> x' (/ y) (* y) (= x')))
+ ))]
+
+ ["Nat" R;nat =+ ++ -+ *+ /+ ;%+ >+ +0 +1 +1000000 %n (;%+ +1000) id]
+ ["Int" R;int = + - * / ;% > 0 1 1000000 %i (;% 1000) id]
+ ["Real" R;real =. +. -. *. /. ;%. >. 0.0 1.0 1000000.0 %r id math;floor]
+ )
+
+(do-template [category rand-gen -> <- = <cap> %a %z]
+ [(test: (format "[" category "] " "Numeric conversions")
+ [value rand-gen
+ #let [value (<cap> value)]]
+ (assert ""
+ (|> value -> <- (= value))))]
+
+ ["Int->Nat" R;int int-to-nat nat-to-int = (;% 1000000) %i %n]
+ ["Nat->Int" R;nat nat-to-int int-to-nat =+ (;%+ +1000000) %n %i]
+ ["Int->Real" R;int int-to-real real-to-int = (;% 1000000) %i %r]
+ ["Real->Int" R;real real-to-int int-to-real =. math;floor %r %i]
+ ## [R;real real-to-frac frac-to-real =. (;%. 1.0) %r %f]
+ )
+
+(test: "Simple macros and constructs"
+ (all (match ["lux" "yolo"] (ident-for ;yolo))
+ (match ["test/lux" "yolo"] (ident-for ;;yolo))
+ (match ["" "yolo"] (ident-for yolo))
+ (match ["lux/test" "yolo"] (ident-for lux/test;yolo))
+ (match ["lux" "yolo"] (ident-for #;yolo))
+ (match ["test/lux" "yolo"] (ident-for #;;yolo))
+ (match ["" "yolo"] (ident-for #yolo))
+ (match ["lux/test" "yolo"] (ident-for #lux/test;yolo))
+
+ (match 1000 (loop [counter 0
+ value 1]
+ (if (< 3 counter)
+ (recur (inc counter) (* 10 value))
+ value)))
+
+ (match (^ (list 1 2 3))
+ (list 1 2 3))
+ (match (^ (list 1 2 3 4 5 6))
+ (list& 1 2 3 (list 4 5 6)))
+
+ (match "yolo" (default "yolo"
+ #;None))
+ (match "lol" (default "yolo"
+ (#;Some "lol")))
+ ))
diff --git a/stdlib/test/test/lux/cli.lux b/stdlib/test/test/lux/cli.lux
new file mode 100644
index 000000000..c95ec9e9c
--- /dev/null
+++ b/stdlib/test/test/lux/cli.lux
@@ -0,0 +1,84 @@
+(;module:
+ [lux #- not]
+ (lux (codata [io])
+ (control monad)
+ (data text/format
+ [number]
+ [product]
+ [sum])
+ (codata function)
+ [cli #- run])
+ [lux/test #- assert])
+
+(test: "lux/cli exports"
+ (test-all (match (#;Right "foo")
+ (cli;run any (list "foo" "bar" "baz")))
+ (match (#;Left _)
+ (cli;run any (list)))
+ (match (#;Right 123)
+ (cli;run (parse (:: number;Codec<Text,Int> decode) any) (list "123")))
+ (match (#;Left _)
+ (cli;run (option (list "-p" "--port")) (list)))
+ (match (#;Left _)
+ (cli;run (option (list "-p" "--port")) (list "yolo")))
+ (match (#;Right "123")
+ (cli;run (option (list "-p" "--port")) (list "-p" "123")))
+ (match (#;Right "123")
+ (cli;run (option (list "-p" "--port")) (list "--port" "123")))
+ (match (#;Right false)
+ (cli;run (flag (list "-h" "--help")) (list)))
+ (match (#;Right false)
+ (cli;run (flag (list "-h" "--help")) (list "yolo")))
+ (match (#;Right true)
+ (cli;run (flag (list "-h" "--help")) (list "-h")))
+ (match (#;Right true)
+ (cli;run (flag (list "-h" "--help")) (list "--help")))
+ (match (#;Right [])
+ (cli;run end (list)))
+ (match (#;Left _)
+ (cli;run end (list "yolo")))
+ (match (#;Left "YOLO")
+ (cli;run (assert false "YOLO") (list "yolo")))
+ (match (#;Right [])
+ (cli;run (assert true "YOLO") (list "yolo")))
+ (match (#;Right #;None)
+ (cli;run (opt any) (list)))
+ (match (#;Right (#;Some "yolo"))
+ (cli;run (opt any) (list "yolo")))
+ (match (#;Right ["foo" "bar"])
+ (cli;run (seq any any) (list "foo" "bar" "baz")))
+ (match (#;Right ["foo" "bar"])
+ (cli;run (seq any any) (list "foo" "bar")))
+ (match (#;Left _)
+ (cli;run (seq any any) (list "foo")))
+ ## (match (#;Right (#;Left 123))
+ ## (cli;run (alt (parse (:: number;Codec<Text,Int> decode) any)
+ ## any)
+ ## (list "123" "foo")))
+ ## (match (#;Right (#;Right "foo"))
+ ## (cli;run (alt (parse (:: number;Codec<Text,Int> decode) any)
+ ## any)
+ ## (list "foo")))
+ (match (#;Left _)
+ (cli;run (alt (parse (:: number;Codec<Text,Int> decode) any)
+ (parse (:: number;Codec<Text,Real> decode) any))
+ (list "foo")))
+ (match (#;Left _)
+ (cli;run (not (parse (:: number;Codec<Text,Int> decode) any))
+ (list "123")))
+ (match (#;Right [])
+ (cli;run (not (parse (:: number;Codec<Text,Int> decode) any))
+ (list "yolo")))
+ (match (^ (#;Right (list "foo" "bar" "baz")))
+ (cli;run (some any) (list "foo" "bar" "baz")))
+ (match (^ (#;Right (list)))
+ (cli;run (some any) (list)))
+ (match (^ (#;Right (list "foo" "bar" "baz")))
+ (cli;run (many any) (list "foo" "bar" "baz")))
+ (match (#;Left _)
+ (cli;run (many any) (list)))
+ (match (#;Right "yolo")
+ (cli;run (either (parse sum;right any)
+ any)
+ (list "yolo")))
+ ))
diff --git a/stdlib/test/test/lux/codata/env.lux b/stdlib/test/test/lux/codata/env.lux
new file mode 100644
index 000000000..7a374cd4d
--- /dev/null
+++ b/stdlib/test/test/lux/codata/env.lux
@@ -0,0 +1,23 @@
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data [text "Text/" Monoid<Text>]
+ text/format
+ [number])
+ (codata function
+ env))
+ lux/test)
+
+(test: "lux/codata/env exports"
+ (test-all (match 123 (run 123 ask))
+ (match 246 (run 123 (local (* 2) ask)))
+ (match 134 (run 123 (:: Functor<Env> map inc (+ 10))))
+ (match 10 (run 123 (:: Applicative<Env> wrap 10)))
+ (match 30 (run 123 (let [(^open) Applicative<Env>]
+ (apply (wrap (+ 10)) (wrap 20)))))
+ (match 30 (run 123 (do Monad<Env>
+ [f (wrap +)
+ x (wrap 10)
+ y (wrap 20)]
+ (wrap (f x y)))))))
diff --git a/stdlib/test/test/lux/codata/io.lux b/stdlib/test/test/lux/codata/io.lux
new file mode 100644
index 000000000..5d521faff
--- /dev/null
+++ b/stdlib/test/test/lux/codata/io.lux
@@ -0,0 +1,21 @@
+(;module:
+ lux
+ (lux (control monad)
+ (data [text "Text/" Monoid<Text>]
+ text/format
+ [number])
+ (codata function
+ io))
+ lux/test)
+
+(test: "lux/codata/io exports"
+ (test-all (match "YOLO" (run (io "YOLO")))
+ (match 11 (run (:: Functor<IO> map inc (io 10))))
+ (match 10 (run (:: Applicative<IO> wrap 10)))
+ (match 30 (run (let [(^open) Applicative<IO>]
+ (apply (wrap (+ 10)) (wrap 20)))))
+ (match 30 (run (do Monad<IO>
+ [f (wrap +)
+ x (wrap 10)
+ y (wrap 20)]
+ (wrap (f x y)))))))
diff --git a/stdlib/test/test/lux/codata/state.lux b/stdlib/test/test/lux/codata/state.lux
new file mode 100644
index 000000000..054b59d45
--- /dev/null
+++ b/stdlib/test/test/lux/codata/state.lux
@@ -0,0 +1,34 @@
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data [text "Text/" Monoid<Text>]
+ text/format
+ [number]
+ [product])
+ (codata function
+ state))
+ lux/test)
+
+(test: "lux/codata/state exports"
+ (test-all (match 123 (product;right (run 123 get)))
+ (match 321 (product;right (run 123 (do Monad<State>
+ [_ (put 321)]
+ get))))
+ (match 369 (product;right (run 123 (do Monad<State>
+ [_ (update (* 3))]
+ get))))
+ (match 124 (product;right (run 123 (use inc))))
+ (match 246 (product;right (run 123 (local (* 2) get))))
+ (match 124 (product;right (run 123 (:: Functor<State> map inc get))))
+ (match 10 (product;right (run 123 (:: Applicative<State> wrap 10))))
+ (match 30 (product;right (run 123 (let [(^open) Applicative<State>]
+ (apply (wrap (+ 10)) (wrap 20))))))
+ (match 30 (product;right (run 123 (: (State Int Int)
+ (do Monad<State>
+ [f (wrap +)
+ x (wrap 10)
+ y (wrap 20)]
+
+ (wrap (f x y)))))))
+ ))
diff --git a/stdlib/test/test/lux/codata/struct/stream.lux b/stdlib/test/test/lux/codata/struct/stream.lux
new file mode 100644
index 000000000..28292a405
--- /dev/null
+++ b/stdlib/test/test/lux/codata/struct/stream.lux
@@ -0,0 +1,68 @@
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad
+ comonad)
+ (data [text "Text/" Monoid<Text>]
+ text/format
+ [number "Int/" Codec<Text,Int>])
+ (codata function
+ [cont]
+ (struct stream)))
+ lux/test)
+
+(test: "lux/codata/stream exports"
+ (let% [<take+drop+split> (do-template [<take> <drop> <split> <arg>]
+ [(match (^ (list 0 1 2))
+ (<take> <arg> (iterate inc 0)))
+ (match (^=> (^stream& w x y z ...)
+ {[w x y z] [3 4 5 6]})
+ (<drop> <arg> (iterate inc 0)))
+ (match (^=> (^ [(list 0 1 2) _stream_])
+ {_stream_ (^stream& w x y z ...)}
+ {[w x y z] [3 4 5 6]})
+ (<split> <arg> (iterate inc 0)))]
+
+ [take drop split +3]
+ [take-while drop-while split-with (< 3)])
+ ]
+ (test-all (match (^=> (^stream& w x y z ...)
+ {[w x y z] [0 1 2 3]})
+ (iterate inc 0))
+ (match (^=> (^stream& w x y z ...)
+ {[w x y z] [0 0 0 0]})
+ (repeat 0))
+ (match (^=> (#;Some the-stream)
+ {the-stream (^stream& w x y z ...)}
+ {[w x y z] [0 1 0 1]})
+ (cycle (list 0 1)))
+ (match 0 (head (iterate inc 0)))
+ (match (^=> (^stream& w x y z ...)
+ {[w x y z] [1 2 3 4]})
+ (tail (iterate inc 0)))
+ (match 9 (at +9 (iterate inc 0)))
+ (match 0 (at +0 (iterate inc 0)))
+ <take+drop+split>
+ (match (^=> (^stream& w x y z ...)
+ {[w x y z] ["0" "1" "2" "3"]})
+ (unfold (lambda [n] [(inc n) (Int/encode n)])
+ 0))
+ (match (^=> (^stream& w x y z ...)
+ {[w x y z] [0 2 4 6]})
+ (filter even? (iterate inc 0)))
+ (match (^=> [e_stream o_stream]
+ {e_stream (^stream& w x y z ...)}
+ {o_stream (^stream& a b c d ...)}
+ {[w x y z a b c d] [0 2 4 6 1 3 5 7]})
+ (partition even? (iterate inc 0)))
+ (match (^=> (^stream& w x y z ...)
+ {[w x y z] [0 1 4 9]})
+ (let [square (lambda [n] (* n n))]
+ (:: Functor<Stream> map square (iterate inc 0))))
+ (match (^=> (^stream& w x y z ...)
+ {[w x y z] [4 9 16 25]})
+ (let [square (lambda [n] (* n n))]
+ (be CoMonad<Stream>
+ [inputs (iterate inc 2)]
+ (square (head inputs)))))
+ )))
diff --git a/stdlib/test/test/lux/concurrency/actor.lux b/stdlib/test/test/lux/concurrency/actor.lux
new file mode 100644
index 000000000..e9a19e8ea
--- /dev/null
+++ b/stdlib/test/test/lux/concurrency/actor.lux
@@ -0,0 +1,70 @@
+(;module:
+ lux
+ (lux (control monad)
+ (data [number]
+ text/format
+ error)
+ (concurrency [promise #+ Promise Monad<Promise> "Promise/" Monad<Promise>]
+ actor)
+ (codata function
+ [io #- run]))
+ lux/test)
+
+(actor: Adder
+ Int
+
+ (method: (add! {offset Int})
+ [Int Int]
+ (let [*state*' (+ offset *state*)]
+ (wrap (#;Right [*state*' [*state* *state*']]))))
+
+ (stop:
+ (exec (log! (format "Cause of death: " (default "???" *cause*)))
+ (log! (format "Current state: " (%i *state*)))
+ (wrap []))))
+
+(test: "lux/concurrency/actor exports"
+ (let [counter-proc (: (Proc Int (Promise Int))
+ [(lambda [self output state]
+ (let [state' (inc state)]
+ (exec (io;run (promise;resolve state' output))
+ (Promise/wrap (#;Right state')))))
+ (lambda [?error state] (Promise/wrap []))])]
+ (test-all (match true
+ (let [counter (: (Actor Int (Promise Int))
+ (io;run (spawn 0 counter-proc)))]
+ (alive? counter)))
+ (match [true false]
+ (let [counter (: (Actor Int (Promise Int))
+ (io;run (spawn 0 counter-proc)))]
+ [(io;run (poison counter))
+ (alive? counter)]))
+ (match [true false]
+ (let [counter (: (Actor Int (Promise Int))
+ (io;run (spawn 0 counter-proc)))]
+ [(io;run (poison counter))
+ (io;run (poison counter))]))
+ (match+ [1 2 3]
+ (do Monad<Promise>
+ [#let [counter (: (Actor Int (Promise Int))
+ (io;run (spawn 0 counter-proc)))
+ output-1 (: (Promise Int) (promise;promise))
+ output-2 (: (Promise Int) (promise;promise))
+ output-3 (: (Promise Int) (promise;promise))]
+ ?1 (send output-1 counter)
+ ?2 (send output-2 counter)
+ ?3 (send output-3 counter)]
+ (if (and ?1 ?2 ?3)
+ (from-promise ($_ promise;seq output-1 output-2 output-3))
+ (wrap (#;Left "Uh, oh...")))))
+ (match+ [[0 1] [1 3] [3 6]]
+ (do Monad<Promise>
+ [#let [adder (: Adder
+ (io;run (spawn 0 Adder//new)))]
+ t1 (add! 1 adder)
+ t2 (add! 2 adder)
+ t3 (add! 3 adder)
+ #let [? (io;run (poison adder))]]
+ (wrap (#;Right [t1 t2 t3]))
+ ))
+ )))
diff --git a/stdlib/test/test/lux/concurrency/frp.lux b/stdlib/test/test/lux/concurrency/frp.lux
new file mode 100644
index 000000000..62ca0b57d
--- /dev/null
+++ b/stdlib/test/test/lux/concurrency/frp.lux
@@ -0,0 +1,54 @@
+(;module:
+ lux
+ (lux (control monad)
+ (data [number]
+ text/format
+ error)
+ (concurrency [promise #+ Promise Monad<Promise> "Promise/" Monad<Promise>]
+ frp)
+ (codata function
+ io))
+ lux/test)
+
+(def: (List->Chan values)
+ (-> (List Int) (Chan Int))
+ (let [_chan (: (Chan Int) (chan))]
+ (run (do Monad<IO>
+ [_ (mapM Monad<IO>
+ (lambda [value]
+ (write value _chan))
+ values)
+ _ (close _chan)]
+ (wrap _chan)))))
+
+(test: "lux/concurrency/frp exports"
+ (test-all (match+ (^ (list 0 1 2 3 4 5))
+ (from-promise (consume (List->Chan (list 0 1 2 3 4 5)))))
+ (match+ (^ (list 0 1 2 3 4 5))
+ (from-promise (consume (let [input (List->Chan (list 0 1 2 3 4 5))
+ output (: (Chan Int) (chan))]
+ (exec (pipe input output)
+ output)))))
+ (match+ (^ (list 0 2 4))
+ (from-promise (consume (filter even? (List->Chan (list 0 1 2 3 4 5))))))
+ (match+ (^ (list 0 1 2 3 4 5 0 -1 -2 -3 -4 -5))
+ (from-promise (consume (merge (list (List->Chan (list 0 1 2 3 4 5))
+ (List->Chan (list 0 -1 -2 -3 -4 -5)))))))
+ (match+ 15 (from-promise (fold (lambda [base input] (Promise/wrap (+ input base))) 0 (List->Chan (list 0 1 2 3 4 5)))))
+ (match+ (^ (list 0 1 2 3 4 5))
+ (from-promise (consume (no-dups number;Eq<Int> (List->Chan (list 0 0 0 1 2 2 3 3 3 3 4 4 4 5 5))))))
+ (match+ (^ (list 12345))
+ (from-promise (consume (as-chan (:: promise;Monad<Promise> wrap 12345)))))
+ (match+ (^ (list 1 2 3 4 5 6))
+ (from-promise (consume (:: Functor<Chan> map inc (List->Chan (list 0 1 2 3 4 5))))))
+ (match+ (^ (list 12345))
+ (from-promise (consume (:: Applicative<Chan> wrap 12345))))
+ (match+ (^ (list 12346))
+ (from-promise (consume (let [(^open) Applicative<Chan>]
+ (apply (wrap inc) (wrap 12345))))))
+ (match+ (^ (list 12346))
+ (from-promise (consume (do Monad<Chan>
+ [f (wrap inc)
+ a (wrap 12345)]
+ (wrap (f a))))))
+ ))
diff --git a/stdlib/test/test/lux/concurrency/promise.lux b/stdlib/test/test/lux/concurrency/promise.lux
new file mode 100644
index 000000000..77e5a0aed
--- /dev/null
+++ b/stdlib/test/test/lux/concurrency/promise.lux
@@ -0,0 +1,31 @@
+(;module:
+ lux
+ (lux (control monad)
+ (data [number]
+ text/format
+ error)
+ (concurrency promise)
+ (codata function
+ [io #*]))
+ lux/test)
+
+(test: "lux/concurrency/promise exports"
+ (test-all (match+ true (from-promise (future (io true))))
+ (match+ [] (from-promise (wait +500)))
+ (match+ [true false] (from-promise (seq (future (io true))
+ (future (io false)))))
+ (match+ (#;Left true) (from-promise (alt (delay +100 true)
+ (delay +200 false))))
+ (match+ (#;Right false) (from-promise (alt (delay +200 true)
+ (delay +100 false))))
+ (match+ true (from-promise (either (delay +100 true)
+ (delay +200 false))))
+ (match+ false (from-promise (either (delay +200 true)
+ (delay +100 false))))
+ (match (#;Some true) (poll (:: Monad<Promise> wrap true)))
+ (match #;None (poll (delay +200 true)))
+ (match false (io;run (resolve false (:: Monad<Promise> wrap true))))
+ (match true (io;run (resolve true (: (Promise Bool) (promise)))))
+ (match+ #;None (from-promise (time-out +100 (delay +200 true))))
+ (match+ (#;Some true) (from-promise (time-out +200 (delay +100 true))))
+ ))
diff --git a/stdlib/test/test/lux/concurrency/stm.lux b/stdlib/test/test/lux/concurrency/stm.lux
new file mode 100644
index 000000000..e29a5294b
--- /dev/null
+++ b/stdlib/test/test/lux/concurrency/stm.lux
@@ -0,0 +1,57 @@
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data [number]
+ (struct [list "" Functor<List>])
+ text/format)
+ (concurrency stm
+ [promise])
+ (codata function))
+ lux/test)
+
+(def: vars Int 5)
+(def: processes/vars Int 5)
+(def: iterations/processes Int 100)
+
+(test: "lux/concurrency/stm exports"
+ (let [_var (var 0)
+ changes (io;run (follow "test" _var))
+ tests (: (List (Test Int))
+ (map (lambda [_]
+ (let [_concurrency-var (var 0)]
+ (from-promise (do promise;Monad<Promise>
+ [_ (seqM @
+ (map (lambda [_]
+ (mapM @ (lambda [_] (commit (update inc _concurrency-var)))
+ (list;range 1 iterations/processes)))
+ (list;range 1 processes/vars)))]
+ (commit (read _concurrency-var))))))
+ (list;range 1 vars)))]
+ (test-all (match+ 0 (commit (do Monad<STM>
+ [value (read _var)]
+ (wrap (#;Right value)))))
+ (match+ 5 (commit (do Monad<STM>
+ [_ (write 5 _var)
+ value (read _var)]
+ (wrap (#;Right value)))))
+ (match+ 5 (commit (do Monad<STM>
+ [value (read _var)]
+ (wrap (#;Right value)))))
+ (match+ 15 (commit (do Monad<STM>
+ [_ (update (* 3) _var)
+ value (read _var)]
+ (wrap (#;Right value)))))
+ (match+ 15 (commit (do Monad<STM>
+ [value (read _var)]
+ (wrap (#;Right value)))))
+ (match+ [5 15] (do promise;Monad<Promise>
+ [?c1+changes' changes
+ #let [[c1 changes'] (default [-1 changes] ?c1+changes')]
+ ?c2+changes' changes'
+ #let [[c2 changes'] (default [-1 changes] ?c2+changes')]]
+ (wrap (#;Right [c1 c2]))))
+ ## Temporarily commented-out due to type-checking bug in
+ ## compiler...
+ ## (match+ _ (seqM Monad<Test> tests))
+ )))
diff --git a/stdlib/test/test/lux/data/bit.lux b/stdlib/test/test/lux/data/bit.lux
new file mode 100644
index 000000000..e20027818
--- /dev/null
+++ b/stdlib/test/test/lux/data/bit.lux
@@ -0,0 +1,65 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (control [monad])
+ (codata [io])
+ (data ["&" bit]
+ number)
+ (math ["R" random]))
+ lux/test)
+
+(def: width Nat +64)
+
+(test: "Bitwise operations."
+ [pattern R;nat
+ idx (:: @ map (%+ width) R;nat)]
+ (all (assert "" (and (<+ (&;count (&;set idx pattern))
+ (&;count (&;clear idx pattern)))
+ (<=+ (&;count pattern)
+ (&;count (&;clear idx pattern)))
+ (>=+ (&;count pattern)
+ (&;count (&;set idx pattern)))
+
+ (or (and (&;set? idx pattern)
+ (not (&;set? idx (&;clear idx pattern))))
+ (and (not (&;set? idx pattern))
+ (&;set? idx (&;set idx pattern))))
+
+ (or (and (&;set? idx pattern)
+ (not (&;set? idx (&;flip idx pattern))))
+ (and (not (&;set? idx pattern))
+ (&;set? idx (&;flip idx pattern))))
+
+ (=+ width
+ (++ (&;count pattern)
+ (&;count (&;~ pattern))))
+
+ (=+ +0
+ (&;& pattern
+ (&;~ pattern)))
+ (=+ (&;~ +0)
+ (&;| pattern
+ (&;~ pattern)))
+ (=+ (&;~ +0)
+ (&;^ pattern
+ (&;~ pattern)))
+ (=+ +0
+ (&;^ pattern
+ pattern))
+
+ (|> pattern (&;rotate-left idx) (&;rotate-right idx) (=+ pattern))
+ (|> pattern (&;rotate-right idx) (&;rotate-left idx) (=+ pattern))
+ (|> pattern (&;rotate-left idx) (&;rotate-left (-+ idx width)) (=+ pattern))
+ (|> pattern (&;rotate-right idx) (&;rotate-right (-+ idx width)) (=+ pattern))
+ ))
+
+ (assert "Shift right respect the sign of ints."
+ (let [value (nat-to-int pattern)]
+ (if (< 0 value)
+ (< 0 (&;>> idx value))
+ (>= 0 (&;>> idx value)))))
+ ))
diff --git a/stdlib/test/test/lux/data/bool.lux b/stdlib/test/test/lux/data/bool.lux
new file mode 100644
index 000000000..218846e2e
--- /dev/null
+++ b/stdlib/test/test/lux/data/bool.lux
@@ -0,0 +1,38 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (control [monad])
+ (codata [io])
+ (data bool)
+ (math ["R" random]))
+ lux/test)
+
+(test: "Boolean operations."
+ [value R;bool]
+ (assert "" (and (not (and value (not value)))
+ (or value (not value))
+
+ (not (:: Or@Monoid<Bool> unit))
+ (:: Or@Monoid<Bool> append value (not value))
+ (:: And@Monoid<Bool> unit)
+ (not (:: And@Monoid<Bool> append value (not value)))
+
+ (:: Eq<Bool> = value (not (not value)))
+ (not (:: Eq<Bool> = value (not value)))
+
+ (not (:: Eq<Bool> = value ((complement id) value)))
+ (:: Eq<Bool> = value ((complement not) value))
+
+ (case (|> value
+ (:: Codec<Text,Bool> encode)
+ (:: Codec<Text,Bool> decode))
+ (#;Right dec-value)
+ (:: Eq<Bool> = value dec-value)
+
+ (#;Left _)
+ false)
+ )))
diff --git a/stdlib/test/test/lux/data/char.lux b/stdlib/test/test/lux/data/char.lux
new file mode 100644
index 000000000..ab2e84d59
--- /dev/null
+++ b/stdlib/test/test/lux/data/char.lux
@@ -0,0 +1,47 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (control [monad])
+ (codata [io])
+ (data char
+ [text])
+ (math ["R" random])
+ pipe
+ [host #- try])
+ lux/test)
+
+(test: "Char operations"
+ [value R;char]
+ (assert "" (and (:: Eq<Char> = value value)
+ (|> value code char (:: Eq<Char> = value))
+ (|> value
+ (:: Codec<Text,Char> encode)
+ (:: Codec<Text,Char> decode)
+ (case> (#;Right dec-value)
+ (:: Eq<Char> = value dec-value)
+
+ (#;Left _)
+ false))
+ (|> value as-text
+ (text;at +0) (default (undefined))
+ (:: Eq<Char> = value))
+ (|> value as-text text;upper-case
+ (text;at +0) (default (undefined))
+ (:: Ord<Char> <= value))
+ (|> value as-text text;lower-case
+ (text;at +0) (default (undefined))
+ (:: Ord<Char> >= value))
+ )))
+
+(test: "Special cases"
+ (all (assert "" (space? #" "))
+ (assert "" (space? #"\n"))
+ (assert "" (space? #"\t"))
+ (assert "" (space? #"\r"))
+ (assert "" (space? #"\f"))
+ (assert "" (not (space? #"a")))
+ ))
diff --git a/stdlib/test/test/lux/data/error.lux b/stdlib/test/test/lux/data/error.lux
new file mode 100644
index 000000000..a1d2cb6ff
--- /dev/null
+++ b/stdlib/test/test/lux/data/error.lux
@@ -0,0 +1,42 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data error))
+ lux/test)
+
+(test: "lux/data/error exports"
+ (all (match (#;Right 11)
+ (:: Functor<Error> map inc (: (Error Int)
+ (#;Right 10))))
+ (match (#;Left "YOLO")
+ (:: Functor<Error> map inc (: (Error Int)
+ (#;Left "YOLO"))))
+
+ (match (#;Right 20)
+ (:: Applicative<Error> wrap 20))
+ (match (#;Right 11)
+ (let [(^open) Applicative<Error>]
+ (apply (wrap inc) (wrap 10))))
+ (match (#;Left "YOLO")
+ (let [(^open) Applicative<Error>]
+ (apply (wrap inc) (#;Left "YOLO"))))
+
+ (match (#;Right 30)
+ (do Monad<Error>
+ [f (wrap +)
+ a (wrap 10)
+ b (wrap 20)]
+ (wrap (f a b))))
+ (match (#;Left "YOLO")
+ (do Monad<Error>
+ [f (wrap +)
+ a (#;Left "YOLO")
+ b (wrap 20)]
+ (wrap (f a b))))
+ ))
diff --git a/stdlib/test/test/lux/data/format/json.lux b/stdlib/test/test/lux/data/format/json.lux
new file mode 100644
index 000000000..78b0b1a76
--- /dev/null
+++ b/stdlib/test/test/lux/data/format/json.lux
@@ -0,0 +1,314 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data [text "Text/" Monoid<Text>]
+ text/format
+ error
+ (format [json #* "JSON/" Eq<JSON> Codec<Text,JSON>])
+ (struct [vector #+ vector]
+ [dict]))
+ [compiler #+ with-gensyms]
+ (macro [ast]
+ [syntax #+ syntax:]
+ [poly #+ derived:])
+ [pipe]
+ test)
+ )
+
+## [Utils]
+(syntax: (reads-to-itself expr)
+ (with-gensyms [g!json g!parsed g!message]
+ (wrap (list (` (: (Test Unit)
+ (let [(~ g!json) (~ expr)]
+ (case (|> (~ g!json) JSON/encode JSON/decode)
+ (#;Left (~ g!message))
+ (fail (~ g!message))
+
+ (#;Right (~ g!parsed))
+ (if (JSON/= (~ g!json) (~ g!parsed))
+ (~ (' (:: Monad<Test> wrap [])))
+ (fail (format "Expression does not parse to itself: " (~ (ast;text (ast;ast-to-text expr)))
+ "\n\nWhich is: " (|> (~ g!json) JSON/encode)
+ "\n\nInstead, it parsed to: " (JSON/encode (~ g!parsed))))
+ ))))
+ )))))
+
+## [Tests]
+## (derived: (Codec<JSON,?> ;Bool))
+## (derived: (Codec<JSON,?> ;Int))
+## (derived: (Codec<JSON,?> ;Real))
+## (derived: (Codec<JSON,?> ;Char))
+## (derived: (Codec<JSON,?> ;Text))
+
+## (type: Int-List (List Int))
+## (derived: (Codec<JSON,?> ;;Int-List))
+
+## (type: Int-Maybe (Maybe Int))
+## (derived: (Codec<JSON,?> ;;Int-Maybe))
+
+## (type: Triple [Bool Int Text])
+## (derived: (Codec<JSON,?> ;;Triple))
+
+## (type: User
+## {#alive? Bool
+## #age Int
+## #name Text})
+## (derived: (Codec<JSON,?> ;;User))
+
+## (type: Options
+## (#One Bool)
+## (#Two Int)
+## (#Three Text))
+## (derived: (Codec<JSON,?> ;;Options))
+
+## (test: "Auto-generated codecs"
+## (let% [<tests> (do-template [<input> <output> <codec>]
+## [(match <output>
+## (|> <input>
+## (:: <codec> encode)
+## JSON/encode))
+## (match+ <input>
+## (should-pass (|> (JSON/decode <output>)
+## (pipe;%> Error/Monad
+## [(:: <codec> decode)]))))]
+
+## [true "true" Codec<JSON,Bool>]
+## [123 "123.0" Codec<JSON,Int>]
+## [123.45 "123.45" Codec<JSON,Real>]
+## [#"a" "\"a\"" Codec<JSON,Char>]
+## ["yolo" "\"yolo\"" Codec<JSON,Text>]
+
+## [(#;Cons 1 (#;Cons 2 (#;Cons 3 #;Nil))) "[1.0,2.0,3.0]" Codec<JSON,Int-List>]
+## [#;Nil "[]" Codec<JSON,Int-List>]
+## [(#;Some 1) "1.0" Codec<JSON,Int-Maybe>]
+## [#;None "null" Codec<JSON,Int-Maybe>]
+## [[false 456 "lol"] "[false,456.0,\"lol\"]" Codec<JSON,Triple>]
+## [{#alive? true #age 25 #name "Eduardo Julian"}
+## "{\"alive?\":true,\"age\":25.0,\"name\":\"Eduardo Julian\"}"
+## Codec<JSON,User>]
+## [(#One true) "[\"One\",true]" Codec<JSON,Options>]
+## [(#Two 123) "[\"Two\",123.0]" Codec<JSON,Options>]
+## [(#Three "yolo") "[\"Three\",\"yolo\"]" Codec<JSON,Options>]
+## )]
+## (test-all <tests>
+## )))
+
+(test: "Basics"
+ (test-all (match #json;Null
+ null)
+
+ (match (#json;Boolean true)
+ (gen-boolean true))
+
+ (match (#json;Boolean false)
+ (gen-boolean false))
+
+ (match (#json;Number 123.45)
+ (gen-number 123.45))
+
+ (match (#json;String "YOLO")
+ (gen-string "YOLO"))
+
+ ## (match (^ (#json;Array (list (#json;Boolean true) (#json;Number 123.45) (#json;String "YOLO"))))
+ ## (json [(gen-boolean true) (gen-number 123.45) (gen-string "YOLO")]))
+
+ ## (match (^ (#json;Object (list ["yolo" (#json;Boolean true)]
+ ## ["lol" (#json;Number 123.45)])))
+ ## (json {"yolo" (gen-boolean true)
+ ## "lol" (gen-number 123.45)}))
+
+ (match (#;Some (#json;Boolean true))
+ (get "yolo" (json {"yolo" true
+ "lol" 123.45})))
+
+ (match (#;Left _)
+ (get "yolo" (json {})))
+
+ ## (match (^ (#;Some (#json;Object (list ["lol" (#json;Number 123.45)]
+ ## ["yolo" (#json;Boolean true)]))))
+ ## (|> (json {"yolo" (gen-boolean true)})
+ ## (set "lol" (gen-number 123.45))))
+
+ (match (#;Right true)
+ (get-boolean "value" (json {"value" true})))
+
+ (match (#;Right 123.45)
+ (get-number "value" (json {"value" 123.45})))
+
+ (match (#;Right "YOLO")
+ (get-string "value" (json {"value" "YOLO"})))
+
+ ## (match (^ (#;Right (list (#json;Boolean true) (#json;Number 123.45) (#json;String "YOLO"))))
+ ## (get-array "value" (json {"value" (json [(gen-boolean true)
+ ## (gen-number 123.45)
+ ## (gen-string "YOLO")])})))
+
+ ## (match (^ (#;Right (list ["yolo" (#json;Boolean true)]
+ ## ["lol" (#json;Number 123.45)])))
+ ## (get-object "value" (json {"value" (json {"yolo" (gen-boolean true)
+ ## "lol" (gen-number 123.45)})})))
+
+ (match (#;Left _)
+ (get-array "value" (json {})))
+
+ (match (#;Left _)
+ (get-array "value" (gen-boolean true)))
+ ))
+
+(test: "Encoding"
+ (test-all (match "null"
+ (JSON/encode (json #null)))
+
+ (match "123.0"
+ (JSON/encode (json 123)))
+
+ (match "123.46"
+ (JSON/encode (json 123.46)))
+
+ (match "true"
+ (JSON/encode (json true)))
+
+ (match "false"
+ (JSON/encode (json false)))
+
+ (match "\"YOLO\""
+ (JSON/encode (json "YOLO")))
+
+ (match "[null,123.46,true,\"YOLO\",[\"nyan\",\"cat\"]]"
+ (JSON/encode (json [#null 123.46 true "YOLO" ["nyan" "cat"]])))
+
+ (match "{\"foo\":\"bar\",\"baz\":null,\"quux\":[\"nyan\",{\"cat\":\"meme\"}]}"
+ (JSON/encode (json {"foo" "bar"
+ "baz" #null
+ "quux" ["nyan" {"cat" "meme"}]})))
+ ))
+
+(test: "Decoding"
+ (test-all (reads-to-itself (json #null))
+ (reads-to-itself (json 123))
+ (reads-to-itself (json 123.46))
+ (reads-to-itself (json true))
+ (reads-to-itself (json false))
+ (reads-to-itself (json "\tY\"OLO\n"))
+ (reads-to-itself (json [#null 123.46 true "YOLO" ["nyan" "cat"]]))
+ (reads-to-itself (json {"foo" "bar"
+ "baz" #null
+ "quux" ["nyan" {"cat" "meme"}]}))
+ ))
+
+(test: "Parser"
+ (test-all (should-pass (run unit
+ (json #null)))
+ (should-fail (run unit
+ (json 123)))
+
+ (match+ 123.45
+ (should-pass (run real
+ (json 123.45))))
+ (should-fail (run real
+ (json #null)))
+
+ (match+ 123
+ (should-pass (run int
+ (json 123))))
+ (should-fail (run int
+ (json #null)))
+
+ (match+ true
+ (should-pass (run bool
+ (json true))))
+ (should-fail (run bool
+ (json 123)))
+
+ (match+ "YOLO"
+ (should-pass (run text
+ (json "YOLO"))))
+ (should-fail (run text
+ (json 123)))
+
+ (match+ (^ (list "YOLO" "LOL" "MEME"))
+ (should-pass (run (array text)
+ (json ["YOLO" "LOL" "MEME"]))))
+ (should-fail (run (array text)
+ (json 123)))
+
+ (match+ "LOL"
+ (should-pass (run (at +1 text)
+ (json ["YOLO" "LOL" "MEME"]))))
+ (should-fail (run (array text)
+ (json 123)))
+
+ (match+ "MEME"
+ (should-pass (run (field "baz" text)
+ (json {"foo" "YOLO"
+ "bar" "LOL"
+ "baz" "MEME"}))))
+ (should-fail (run (field "baz" text)
+ (json 123)))
+
+ (match+ (#json;Number 123.0)
+ (should-pass (run any
+ (json 123))))
+
+ (match+ ["YOLO" "MEME"]
+ (should-pass (run (seq (field "foo" text)
+ (field "baz" text))
+ (json {"foo" "YOLO"
+ "bar" "LOL"
+ "baz" "MEME"}))))
+ (should-fail (run (seq (field "foo" text)
+ (field "baz" text))
+ (json {"foo" "YOLO"
+ "bar" "LOL"})))
+
+ (match+ (#;Left "YOLO")
+ (should-pass (run (alt (field "foo" text)
+ (field "baz" text))
+ (json {"foo" "YOLO"
+ "bar" "LOL"
+ "baz" "MEME"}))))
+ (match+ (#;Right "MEME")
+ (should-pass (run (alt (field "fool" text)
+ (field "baz" text))
+ (json {"foo" "YOLO"
+ "bar" "LOL"
+ "baz" "MEME"}))))
+ (should-fail (run (alt (field "fool" text)
+ (field "baz" text))
+ (json {"foo" "YOLO"
+ "bar" "LOL"})))
+
+ (match+ "YOLO"
+ (should-pass (run (either (field "foo" text)
+ (field "baz" text))
+ (json {"foo" "YOLO"
+ "bar" "LOL"
+ "baz" "MEME"}))))
+ (match+ "MEME"
+ (should-pass (run (either (field "fool" text)
+ (field "baz" text))
+ (json {"foo" "YOLO"
+ "bar" "LOL"
+ "baz" "MEME"}))))
+ (should-fail (run (either (field "fool" text)
+ (field "baz" text))
+ (json {"foo" "YOLO"
+ "bar" "LOL"})))
+
+ (match+ (#;Some "YOLO")
+ (should-pass (run (opt (field "foo" text))
+ (json {"foo" "YOLO"
+ "bar" "LOL"
+ "baz" "MEME"}))))
+ (match+ #;None
+ (should-pass (run (opt (field "fool" text))
+ (json {"foo" "YOLO"
+ "bar" "LOL"
+ "baz" "MEME"}))))
+ ))
diff --git a/stdlib/test/test/lux/data/ident.lux b/stdlib/test/test/lux/data/ident.lux
new file mode 100644
index 000000000..8cb85175f
--- /dev/null
+++ b/stdlib/test/test/lux/data/ident.lux
@@ -0,0 +1,53 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data ["&" ident]
+ [text "Text/" Eq<Text>])
+ (math ["R" random])
+ pipe)
+ lux/test)
+
+(test: "Idents"
+ [## First Ident
+ sizeM1 (|> R;nat (:: @ map (%+ +100)))
+ sizeN1 (|> R;nat (:: @ map (%+ +100)))
+ module1 (R;text sizeM1)
+ name1 (R;text sizeN1)
+ #let [ident1 [module1 name1]]
+ ## Second Ident
+ sizeM2 (|> R;nat (:: @ map (%+ +100)))
+ sizeN2 (|> R;nat (:: @ map (%+ +100)))
+ module2 (R;text sizeM2)
+ name2 (R;text sizeN2)
+ #let [ident2 [module2 name2]]
+ #let [(^open "&/") &;Eq<Ident>
+ (^open "&/") &;Codec<Text,Ident>]]
+ (all (assert "Can get the module & name parts of an ident."
+ (and (== module1 (&;module ident1))
+ (== name1 (&;name ident1))))
+
+ (assert "Can compare idents for equality."
+ (and (&/= ident1 ident1)
+ (if (&/= ident1 ident2)
+ (and (Text/= module1 module2)
+ (Text/= name1 name2))
+ (or (not (Text/= module1 module2))
+ (not (Text/= name1 name2))))))
+
+ (assert "Can encode idents as text."
+ (|> ident1
+ &/encode &/decode
+ (case> (#;Right dec-ident) (&/= ident1 dec-ident)
+ _ false)))
+
+ (assert "Encoding an ident without a module component results in text equal to the name of the ident."
+ (if (text;empty? module1)
+ (Text/= name1 (&/encode ident1))
+ true))
+ ))
diff --git a/stdlib/test/test/lux/data/identity.lux b/stdlib/test/test/lux/data/identity.lux
new file mode 100644
index 000000000..f492a801e
--- /dev/null
+++ b/stdlib/test/test/lux/data/identity.lux
@@ -0,0 +1,36 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad
+ comonad)
+ (data identity
+ [text "Text/" Monoid<Text>]))
+ lux/test)
+
+(test: "lux/data/identity exports"
+ (all (match "yololol" (:: Functor<Identity> map (Text/append "yolo") "lol"))
+
+ (match "yolo" (:: Applicative<Identity> wrap "yolo"))
+ (match "yololol" (let [(^open) Applicative<Identity>]
+ (apply (wrap (Text/append "yolo")) (wrap "lol"))))
+
+ (match "yololol"
+ (do Monad<Identity>
+ [f (wrap Text/append)
+ a (wrap "yolo")
+ b (wrap "lol")]
+ (wrap (f a b))))
+
+ (match "yololol" (:: CoMonad<Identity> unwrap "yololol"))
+ (match "yololol"
+ (be CoMonad<Identity>
+ [f Text/append
+ a "yolo"
+ b "lol"]
+ (f a b)))
+ ))
diff --git a/stdlib/test/test/lux/data/log.lux b/stdlib/test/test/lux/data/log.lux
new file mode 100644
index 000000000..c052a29da
--- /dev/null
+++ b/stdlib/test/test/lux/data/log.lux
@@ -0,0 +1,32 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data log
+ [text "Text/" Monoid<Text>]
+ [number])
+ (codata function))
+ lux/test)
+
+(test: "lux/data/log exports"
+ (all (match ["" 11]
+ (:: Functor<Log> map inc ["" 10]))
+ (match ["" 20]
+ (:: (Applicative<Log> text;Monoid<Text>) wrap 20))
+ (match ["" 30]
+ (let [(^open) (Applicative<Log> text;Monoid<Text>)]
+ (apply (wrap (+ 10)) (wrap 20))))
+ (match ["" 30]
+ (do (Monad<Log> text;Monoid<Text>)
+ [f (wrap +)
+ a (wrap 10)
+ b (wrap 20)]
+ (wrap (f a b))))
+ (match ["YOLO" []]
+ (log "YOLO"))
+ ))
diff --git a/stdlib/test/test/lux/data/maybe.lux b/stdlib/test/test/lux/data/maybe.lux
new file mode 100644
index 000000000..bd44593d7
--- /dev/null
+++ b/stdlib/test/test/lux/data/maybe.lux
@@ -0,0 +1,49 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data maybe
+ [text "Text/" Monoid<Text>]
+ [number]))
+ lux/test)
+
+(test: "lux/data/maybe exports"
+ (all (match #;None (:: Monoid<Maybe> unit))
+ (match (#;Some "yolo") (:: Monoid<Maybe> append (#;Some "yolo") (#;Some "lol")))
+ (match (#;Some "yolo") (:: Monoid<Maybe> append (#;Some "yolo") #;None))
+ (match (#;Some "lol") (:: Monoid<Maybe> append #;None (#;Some "lol")))
+ (match #;None (: (Maybe Text) (:: Monoid<Maybe> append #;None #;None)))
+
+ (match #;None (:: Functor<Maybe> map (Text/append "yolo") #;None))
+ (match (#;Some "yololol") (:: Functor<Maybe> map (Text/append "yolo") (#;Some "lol")))
+
+ (match (#;Some "yolo") (:: Applicative<Maybe> wrap "yolo"))
+ (match (#;Some "yololol")
+ (let [(^open) Applicative<Maybe>]
+ (apply (wrap (Text/append "yolo")) (wrap "lol"))))
+
+ (match (#;Some "yololol")
+ (do Monad<Maybe>
+ [f (wrap Text/append)
+ a (wrap "yolo")
+ b (wrap "lol")]
+ (wrap (f a b))))
+
+ (match true (:: (Eq<Maybe> text;Eq<Text>) =
+ (: (Maybe Text) #;None)
+ (: (Maybe Text) #;None)))
+ (match true (:: (Eq<Maybe> text;Eq<Text>) =
+ (#;Some "yolo")
+ (#;Some "yolo")))
+ (match false (:: (Eq<Maybe> text;Eq<Text>) =
+ (#;Some "yolo")
+ (#;Some "lol")))
+ (match false (:: (Eq<Maybe> text;Eq<Text>) =
+ (#;Some "yolo")
+ (: (Maybe Text) #;None)))
+ ))
diff --git a/stdlib/test/test/lux/data/number.lux b/stdlib/test/test/lux/data/number.lux
new file mode 100644
index 000000000..adefb480a
--- /dev/null
+++ b/stdlib/test/test/lux/data/number.lux
@@ -0,0 +1,135 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data number
+ [text "Text/" Monoid<Text>]
+ text/format)
+ (math ["R" random])
+ pipe)
+ lux/test)
+
+(do-template [category rand-gen <Eq> <Ord>]
+ [(test: (format "[" category "] " "Eq & Ord")
+ [x rand-gen
+ y rand-gen]
+ (assert "" (and (:: <Eq> = x x)
+ (or (:: <Eq> = x y)
+ (:: <Ord> < y x)
+ (:: <Ord> > y x)))))]
+
+ ["Nat" R;nat Eq<Nat> Ord<Nat>]
+ ["Int" R;int Eq<Int> Ord<Int>]
+ ["Real" R;real Eq<Real> Ord<Real>]
+ ["Frac" R;frac Eq<Frac> Ord<Frac>]
+ )
+
+(do-template [category rand-gen <Number>]
+ [(test: (format "[" category "] " "Number")
+ [x rand-gen]
+ (assert "" (let [(^open) <Number>]
+ (and (>= x (abs x))
+ (<= x (negate (abs x)))
+ (= x (* (signum x)
+ (abs x)))))))]
+
+ ["Nat" R;nat Number<Nat>]
+ ["Int" R;int Number<Int>]
+ ["Real" R;real Number<Real>]
+ )
+
+(do-template [category rand-gen <Enum> <Number>]
+ [(test: (format "[" category "] " "Enum")
+ [x rand-gen]
+ (assert "" (let [(^open) <Number>]
+ (and (> x
+ (:: <Enum> succ x))
+ (< x
+ (:: <Enum> pred x))
+
+ (= x
+ (|> x (:: <Enum> pred) (:: <Enum> succ)))
+ (= x
+ (|> x (:: <Enum> succ) (:: <Enum> pred)))
+ ))))]
+
+ ["Nat" R;nat Enum<Nat> Number<Nat>]
+ ["Int" R;int Enum<Int> Number<Int>]
+ )
+
+(do-template [category rand-gen <Number> <Bounded>]
+ [(test: (format "[" category "] " "Bounded")
+ [x rand-gen]
+ (assert "" (let [(^open) <Number>]
+ (and (<= x (:: <Bounded> bottom))
+ (>= x (:: <Bounded> top))
+ ))))]
+
+ ["Nat" R;nat Number<Nat> Bounded<Nat>]
+ ["Int" R;int Number<Int> Bounded<Int>]
+ ["Real" R;real Number<Real> Bounded<Real>]
+ )
+
+(do-template [category rand-gen <Number> <Monoid> <cap>]
+ [(test: (format "[" category "] " "Monoid")
+ [x (:: @ map (|>. (:: <Number> abs) <cap>) rand-gen)]
+ (assert "" (let [(^open) <Number>
+ (^open) <Monoid>]
+ (and (= x (append unit x))
+ (= x (append x unit))
+ (= unit (append unit unit))
+ (>= x (append x x))))))]
+
+ ["Nat/Add" R;nat Number<Nat> Add@Monoid<Nat> (;%+ +1000)]
+ ["Nat/Mul" R;nat Number<Nat> Mul@Monoid<Nat> (;%+ +1000)]
+ ["Nat/Min" R;nat Number<Nat> Min@Monoid<Nat> (;%+ +1000)]
+ ["Nat/Max" R;nat Number<Nat> Max@Monoid<Nat> (;%+ +1000)]
+ ["Int/Add" R;int Number<Int> Add@Monoid<Int> (;% 1000)]
+ ["Int/Mul" R;int Number<Int> Mul@Monoid<Int> (;% 1000)]
+ ["Int/Min" R;int Number<Int> Min@Monoid<Int> (;% 1000)]
+ ["Int/Max" R;int Number<Int> Max@Monoid<Int> (;% 1000)]
+ ["Real/Add" R;real Number<Real> Add@Monoid<Real> (;%. 1000.0)]
+ ["Real/Mul" R;real Number<Real> Mul@Monoid<Real> (;%. 1000.0)]
+ ["Real/Min" R;real Number<Real> Min@Monoid<Real> (;%. 1000.0)]
+ ["Real/Max" R;real Number<Real> Max@Monoid<Real> (;%. 1000.0)]
+ )
+
+(do-template [category rand-gen <Number> <Codec>]
+ [(test: (format "[" category "] " "Codec")
+ [x rand-gen]
+ (assert "" (|> x
+ (:: <Codec> encode)
+ (:: <Codec> decode)
+ (case> (#;Right x')
+ (:: <Number> = x x')
+
+ (#;Left _)
+ false))))]
+
+ ["Nat" R;nat Number<Nat> Codec<Text,Nat>]
+ ["Int" R;int Number<Int> Codec<Text,Int>]
+ ["Real" R;real Number<Real> Codec<Text,Real>]
+ ## ["Frac" R;frac Number<Frac> Codec<Text,Frac>]
+ )
+
+(do-template [category rand-gen <Number> <Codec>]
+ [(test: (format "[" category "] " "Alternative formats")
+ [x rand-gen]
+ (assert "" (|> x
+ (:: <Codec> encode)
+ (:: <Codec> decode)
+ (case> (#;Right x')
+ (:: <Number> = x x')
+
+ (#;Left _)
+ false))))]
+
+ ["Nat/Binary" R;nat Number<Nat> Binary@Codec<Text,Nat>]
+ ["Nat/Octal" R;nat Number<Nat> Octal@Codec<Text,Nat>]
+ ["Nat/Hex" R;nat Number<Nat> Hex@Codec<Text,Nat>]
+ )
diff --git a/stdlib/test/test/lux/data/product.lux b/stdlib/test/test/lux/data/product.lux
new file mode 100644
index 000000000..51c23e47d
--- /dev/null
+++ b/stdlib/test/test/lux/data/product.lux
@@ -0,0 +1,20 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data product
+ [text "Text/" Monoid<Text>]
+ [number])
+ (codata function))
+ lux/test)
+
+(test: "Product operations"
+ (all (match 1 (left [1 2]))
+ (match 2 (right [1 2]))
+ (match [2 1] (swap [1 2]))
+ ))
diff --git a/stdlib/test/test/lux/data/struct/array.lux b/stdlib/test/test/lux/data/struct/array.lux
new file mode 100644
index 000000000..171631bd9
--- /dev/null
+++ b/stdlib/test/test/lux/data/struct/array.lux
@@ -0,0 +1,130 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (control [monad])
+ (codata [io])
+ (data (struct ["&" array]
+ [list])
+ [number])
+ (math ["R" random])
+ pipe)
+ lux/test)
+
+(def: bounded-size
+ (R;Random Nat)
+ (|> R;nat
+ (:: R;Monad<Random> map (|>. (%+ +100) (++ +1)))))
+
+(test: "Arrays and their copies"
+ [size bounded-size
+ original (R;array size R;nat)
+ #let [clone (&;clone original)
+ copy (: (&;Array Nat)
+ (&;new size))
+ manual-copy (: (&;Array Nat)
+ (&;new size))]]
+ (all (assert "Size function must correctly return size of array."
+ (=+ size (&;size original)))
+ (assert "Cloning an array should yield and identical array, but not the same one."
+ (and (:: (&;Eq<Array> number;Eq<Nat>) = original clone)
+ (not (== original clone))))
+ (assert "Full-range manual copies should give the same result as cloning."
+ (exec (&;copy size +0 original +0 copy)
+ (and (:: (&;Eq<Array> number;Eq<Nat>) = original copy)
+ (not (== original copy)))))
+ (assert "Array folding should go over all values."
+ (exec (:: &;Fold<Array> fold
+ (lambda [x idx]
+ (exec (&;put idx x manual-copy)
+ (inc+ idx)))
+ +0
+ original)
+ (:: (&;Eq<Array> number;Eq<Nat>) = original manual-copy)))
+ (assert "Transformations between (full) arrays and lists shouldn't cause lose or change any values."
+ (|> original
+ &;to-list &;from-list
+ (:: (&;Eq<Array> number;Eq<Nat>) = original)))
+ ))
+
+(test: "Array mutation"
+ [size bounded-size
+ idx (:: @ map (%+ size) R;nat)
+ array (|> (R;array size R;nat)
+ (R;filter (|>. &;to-list (list;any? odd?+))))
+ #let [value (default (undefined)
+ (&;get idx array))]]
+ (all (assert "Shouldn't be able to find a value in an unoccupied cell."
+ (case (&;get idx (&;remove idx array))
+ (#;Some _) false
+ #;None true))
+ (assert "You should be able to access values put into the array."
+ (case (&;get idx (&;put idx value array))
+ (#;Some value') (=+ value' value)
+ #;None false))
+ (assert "All cells should be occupied on a full array."
+ (and (=+ size (&;occupied array))
+ (=+ +0 (&;vacant array))))
+ (assert "Filtering mutates the array to remove invalid values."
+ (exec (&;filter even?+ array)
+ (and (<+ size (&;occupied array))
+ (>+ +0 (&;vacant array))
+ (=+ size (++ (&;occupied array)
+ (&;vacant array))))))
+ ))
+
+(test: "Finding values."
+ [size bounded-size
+ array (|> (R;array size R;nat)
+ (R;filter (|>. &;to-list (list;any? even?+))))]
+ (all (assert "Can find values inside arrays."
+ (|> (&;find even?+ array)
+ (case> (#;Some _) true
+ #;None false)))
+ (assert "Can find values inside arrays (with access to indices)."
+ (|> (&;find+ (lambda [idx n]
+ (and (even?+ n)
+ (<+ size idx)))
+ array)
+ (case> (#;Some _) true
+ #;None false)))))
+
+(test: "Functor"
+ [size bounded-size
+ array (R;array size R;nat)]
+ (let [(^open) &;Functor<Array>
+ (^open) (&;Eq<Array> number;Eq<Nat>)]
+ (all (assert "Functor shouldn't alter original array."
+ (let [copy (map id array)]
+ (and (= array copy)
+ (not (== array copy)))))
+ (assert "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)))))))
+
+(test: "Monoid"
+ [sizeL bounded-size
+ sizeR bounded-size
+ left (R;array sizeL R;nat)
+ right (R;array sizeR R;nat)
+ #let [(^open) &;Monoid<Array>
+ (^open) (&;Eq<Array> number;Eq<Nat>)
+ fusion (append left right)]]
+ (all (assert "Appending two arrays should produce a new one twice as large."
+ (=+ (++ sizeL sizeR) (&;size fusion)))
+ (assert "First elements of fused array should equal the first array."
+ (|> (: (&;Array Nat)
+ (&;new sizeL))
+ (&;copy sizeL +0 fusion +0)
+ (= left)))
+ (assert "Last elements of fused array should equal the second array."
+ (|> (: (&;Array Nat)
+ (&;new sizeR))
+ (&;copy sizeR sizeL fusion +0)
+ (= right)))
+ ))
diff --git a/stdlib/test/test/lux/data/struct/dict.lux b/stdlib/test/test/lux/data/struct/dict.lux
new file mode 100644
index 000000000..06b9550aa
--- /dev/null
+++ b/stdlib/test/test/lux/data/struct/dict.lux
@@ -0,0 +1,136 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad
+ [eq])
+ (data [text "Text/" Monoid<Text>]
+ text/format
+ [number]
+ [char]
+ (struct ["&" dict]
+ [list "List/" Fold<List> Functor<List>]))
+ (codata function)
+ (math ["R" random])
+ pipe)
+ lux/test)
+
+(test: "Dictionaries."
+ [#let [capped-nat (:: R;Monad<Random> map (%+ +100) R;nat)]
+ size capped-nat
+ dict (R;dict char;Hash<Char> size R;char capped-nat)
+ non-key (|> R;char
+ (R;filter (lambda [key] (not (&;contains? key dict)))))
+ test-val (|> R;nat
+ (R;filter (lambda [val] (not (list;member? number;Eq<Nat> (&;values dict) val)))))]
+ (all (assert "Size function should correctly represent Dict size."
+ (=+ size (&;size dict)))
+
+ (assert "Dicts of size 0 should be considered empty."
+ (if (=+ +0 size)
+ (&;empty? dict)
+ (not (&;empty? dict))))
+
+ (assert "The functions 'entries', 'keys' and 'values' should be synchronized."
+ (:: (list;Eq<List> (eq;conj char;Eq<Char> number;Eq<Nat>)) =
+ (&;entries dict)
+ (list;zip2 (&;keys dict)
+ (&;values dict))))
+
+ (assert "Dict should be able to recognize it's own keys."
+ (list;every? (lambda [key] (&;contains? key dict))
+ (&;keys dict)))
+
+ (assert "Should be able to get every key."
+ (list;every? (lambda [key] (case (&;get key dict)
+ (#;Some _) true
+ _ false))
+ (&;keys dict)))
+
+ (assert "Shouldn't be able to access non-existant keys."
+ (case (&;get non-key dict)
+ (#;Some _) false
+ _ true))
+
+ (assert "Should be able to put and then get a value."
+ (case (&;get non-key (&;put non-key test-val dict))
+ (#;Some v) (=+ test-val v)
+ _ true))
+
+ (assert "Should be able to put~ and then get a value."
+ (case (&;get non-key (&;put~ non-key test-val dict))
+ (#;Some v) (=+ test-val v)
+ _ true))
+
+ (assert "Shouldn't be able to put~ an existing key."
+ (or (=+ +0 size)
+ (let [first-key (|> dict &;keys list;head (default (undefined)))]
+ (case (&;get first-key (&;put~ first-key test-val dict))
+ (#;Some v) (not (=+ test-val v))
+ _ true))))
+
+ (assert "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))))))
+
+ (assert "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)]
+ (=+ (inc+ x) y)
+
+ _
+ false)))
+
+ (assert "Additions and removals to a Dict should affect its size."
+ (let [plus (&;put non-key test-val dict)
+ base (&;remove non-key plus)]
+ (and (=+ (inc+ (&;size dict)) (&;size plus))
+ (=+ (dec+ (&;size plus)) (&;size base)))))
+
+ (assert "A Dict should equal itself & going to<->from lists shouldn't change that."
+ (let [(^open) (&;Eq<Dict> number;Eq<Nat>)]
+ (and (= dict dict)
+ (|> dict &;entries (&;from-list char;Hash<Char>) (= dict)))))
+
+ (assert "Merging a Dict to itself changes nothing."
+ (let [(^open) (&;Eq<Dict> number;Eq<Nat>)]
+ (= dict (&;merge dict dict))))
+
+ (assert "If you merge, and the second dict has overlapping keys, it should overwrite yours."
+ (let [dict' (|> dict &;entries
+ (List/map (lambda [[k v]] [k (inc+ v)]))
+ (&;from-list char;Hash<Char>))
+ (^open) (&;Eq<Dict> number;Eq<Nat>)]
+ (= dict' (&;merge dict' dict))))
+
+ (assert "Can merge values in such a way that they become combined."
+ (list;every? (lambda [[x x*2]] (=+ (*+ +2 x) x*2))
+ (list;zip2 (&;values dict)
+ (&;values (&;merge-with ++ dict dict)))))
+
+ (assert "Should be able to select subset of keys from dict."
+ (|> dict
+ (&;put non-key test-val)
+ (&;select (list non-key))
+ &;size
+ (=+ +1)))
+
+ (assert "Should be able to re-bind existing values to different keys."
+ (or (=+ +0 size)
+ (let [first-key (|> dict &;keys list;head (default (undefined)))
+ rebound (&;re-bind first-key non-key dict)]
+ (and (=+ (&;size dict) (&;size rebound))
+ (&;contains? non-key rebound)
+ (not (&;contains? first-key rebound))
+ (=+ (default (undefined)
+ (&;get first-key dict))
+ (default (undefined)
+ (&;get non-key rebound)))))))
+ ))
diff --git a/stdlib/test/test/lux/data/struct/list.lux b/stdlib/test/test/lux/data/struct/list.lux
new file mode 100644
index 000000000..6baf13c6c
--- /dev/null
+++ b/stdlib/test/test/lux/data/struct/list.lux
@@ -0,0 +1,191 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data (struct ["&" list])
+ [text "Text/" Monoid<Text>]
+ [number]
+ [bool]
+ [product])
+ (math ["R" random])
+ pipe)
+ lux/test)
+
+(def: bounded-size
+ (R;Random Nat)
+ (|> R;nat
+ (:: R;Monad<Random> map (|>. (%+ +100) (++ +10)))))
+
+(test: "Lists"
+ [size bounded-size
+ idx (:: @ map (%+ 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) (&;Eq<List> number;Eq<Nat>)
+ (^open "&/") &;Functor<List>]]
+ (all (assert "The size function should correctly portray the size of the list."
+ (=+ size (&;size sample)))
+
+ (assert "The repeat function should produce as many elements as asked of it."
+ (=+ size (&;size (&;repeat size []))))
+
+ (assert "Reversing a list does not change it's size."
+ (=+ (&;size sample)
+ (&;size (&;reverse sample))))
+
+ (assert "Reversing a list twice results in the original list."
+ (= sample
+ (&;reverse (&;reverse sample))))
+
+ (assert "Filtering by a predicate and its complement should result in a number of elements equal to the original list."
+ (and (=+ (&;size sample)
+ (++ (&;size (&;filter even?+ sample))
+ (&;size (&;filter (bool;complement even?+) sample))))
+ (let [[plus minus] (&;partition even?+ sample)]
+ (=+ (&;size sample)
+ (++ (&;size plus)
+ (&;size minus))))))
+
+ (assert "If every element in a list satisfies a predicate, there can't be any that satisfy its complement."
+ (if (&;every? even?+ sample)
+ (and (not (&;any? (bool;complement even?+) sample))
+ (&;empty? (&;filter (bool;complement even?+) sample)))
+ (&;any? (bool;complement even?+) sample)))
+
+ (assert "Any element of the list can be considered it's member."
+ (let [elem (default (undefined)
+ (&;at idx sample))]
+ (&;member? number;Eq<Nat> sample elem)))
+
+ (assert "Appending the head and the tail should yield the original list."
+ (let [head (default (undefined)
+ (&;head sample))
+ tail (default (undefined)
+ (&;tail sample))]
+ (= sample
+ (#;Cons head tail))))
+
+ (assert "Appending the inits and the last should yield the original list."
+ (let [(^open) &;Monoid<List>
+ inits (default (undefined)
+ (&;inits sample))
+ last (default (undefined)
+ (&;last sample))]
+ (= sample
+ (append inits (list last)))))
+
+ (assert "Functor should go over every element of the list."
+ (let [(^open) &;Functor<List>
+ there (map inc+ sample)
+ back-again (map dec+ there)]
+ (and (not (= sample there))
+ (= sample back-again))))
+
+ (assert "Splitting a list into chunks and re-appending them should yield the original list."
+ (let [(^open) &;Monoid<List>
+ [left right] (&;split idx sample)
+ [left' right'] (&;split-with even?+ sample)]
+ (and (= sample
+ (append left right))
+ (= sample
+ (append left' right'))
+ (= sample
+ (append (&;take idx sample)
+ (&;drop idx sample)))
+ (= sample
+ (append (&;take-while even?+ sample)
+ (&;drop-while even?+ sample)))
+ )))
+
+ (assert "Segmenting the list in pairs should yield as many elements as N/2."
+ (=+ (/+ +2 size)
+ (&;size (&;as-pairs sample))))
+
+ (assert "Sorting a list shouldn't change it's size."
+ (=+ (&;size sample)
+ (&;size (&;sort <+ sample))))
+
+ (assert "Sorting a list with one order should yield the reverse of sorting it with the opposite order."
+ (= (&;sort <+ sample)
+ (&;reverse (&;sort >+ sample))))
+
+ (assert "If you zip 2 lists, the result's size will be that of the smaller list."
+ (=+ (&;size (&;zip2 sample other-sample))
+ (min+ (&;size sample) (&;size other-sample))))
+
+ (assert "I can pair-up elements of a list in order."
+ (let [(^open) &;Functor<List>
+ 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))))))
+
+ (assert "You can generate indices for any size, and they will be in ascending order."
+ (let [(^open) &;Functor<List>
+ indices (&;indices size)]
+ (and (=+ size (&;size indices))
+ (= indices
+ (&;sort <+ indices))
+ (&;every? (=+ (dec+ size))
+ (&;zip2-with ++
+ indices
+ (&;sort >+ indices)))
+ )))
+
+ (assert "The 'interpose' function places a value between every member of a list."
+ (let [(^open) &;Functor<List>
+ sample+ (&;interpose separator sample)]
+ (and (=+ (|> size (*+ +2) dec+)
+ (&;size sample+))
+ (|> sample+ &;as-pairs (map product;right) (&;every? (=+ separator))))))
+
+ (assert "List append is a monoid."
+ (let [(^open) &;Monoid<List>]
+ (and (= sample (append unit sample))
+ (= sample (append sample unit))
+ (let [[left right] (&;split size (append sample other-sample))]
+ (and (= sample left)
+ (= other-sample right))))))
+
+ (assert "Applicative allows you to create singleton lists, and apply lists of functions to lists of values."
+ (let [(^open) &;Applicative<List>]
+ (and (= (list separator) (wrap separator))
+ (= (map inc+ sample)
+ (apply (wrap inc+) sample)))))
+
+ (assert "List concatenation is a monad."
+ (let [(^open) &;Monad<List>
+ (^open) &;Monoid<List>]
+ (= (append sample other-sample)
+ (join (list sample other-sample)))))
+
+ (assert "You can find any value that satisfies some criterium, if such values exist in the list."
+ (case (&;find even?+ sample)
+ (#;Some found)
+ (and (even?+ found)
+ (&;any? even?+ sample)
+ (not (&;every? (bool;complement even?+) sample)))
+
+ #;None
+ (and (not (&;any? even?+ sample))
+ (&;every? (bool;complement even?+) sample))))
+
+ (assert "You can iteratively construct a list, generating values until you're done."
+ (= (&;range+ +0 (dec+ size))
+ (&;iterate (lambda [n] (if (<+ size n) (#;Some (inc+ n)) #;None))
+ +0)))
+
+ (assert "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)))))
+ ))
diff --git a/stdlib/test/test/lux/data/struct/queue.lux b/stdlib/test/test/lux/data/struct/queue.lux
new file mode 100644
index 000000000..895929ab4
--- /dev/null
+++ b/stdlib/test/test/lux/data/struct/queue.lux
@@ -0,0 +1,54 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data (struct ["&" queue])
+ [number])
+ (math ["R" random])
+ pipe)
+ lux/test)
+
+(test: "Queues"
+ [size (:: @ map (%+ +100) R;nat)
+ sample (R;queue size R;nat)
+ non-member (|> R;nat
+ (R;filter (. not (&;enqueued? number;Eq<Nat> sample))))]
+ (all (assert "I can query the size of a queue (and empty queues have size 0)."
+ (if (=+ +0 size)
+ (&;empty? sample)
+ (=+ size (&;size sample))))
+
+ (assert "Enqueueing and dequeing affects the size of queues."
+ (and (=+ (inc+ size) (&;size (&;enqueue non-member sample)))
+ (or (&;empty? sample)
+ (=+ (dec+ size) (&;size (&;dequeue sample))))
+ (=+ size (&;size (&;dequeue (&;enqueue non-member sample))))))
+
+ (assert "Transforming to/from list can't change the queue."
+ (let [(^open "&/") (&;Eq<Queue> number;Eq<Nat>)]
+ (|> sample
+ &;to-list &;from-list
+ (&/= sample))))
+
+ (assert "I can always peek at a non-empty queue."
+ (case (&;peek sample)
+ #;None (&;empty? sample)
+ (#;Some _) true))
+
+ (assert "I can query whether an element belongs to a queue."
+ (and (not (&;enqueued? number;Eq<Nat> sample non-member))
+ (&;enqueued? number;Eq<Nat> (&;enqueue non-member sample)
+ non-member)
+ (case (&;peek sample)
+ #;None
+ (&;empty? sample)
+
+ (#;Some first)
+ (and (&;enqueued? number;Eq<Nat> sample first)
+ (not (&;enqueued? number;Eq<Nat> (&;dequeue sample) first))))))
+ ))
diff --git a/stdlib/test/test/lux/data/struct/set.lux b/stdlib/test/test/lux/data/struct/set.lux
new file mode 100644
index 000000000..3725e7f93
--- /dev/null
+++ b/stdlib/test/test/lux/data/struct/set.lux
@@ -0,0 +1,67 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data (struct ["&" set]
+ [list "" Fold<List>])
+ [number])
+ (math ["R" random])
+ pipe)
+ lux/test)
+
+(def: gen-nat
+ (R;Random Nat)
+ (|> R;nat
+ (:: R;Monad<Random> map (%+ +100))))
+
+(test: "Sets"
+ [sizeL gen-nat
+ sizeR gen-nat
+ setL (R;set number;Hash<Nat> sizeL gen-nat)
+ setR (R;set number;Hash<Nat> sizeR gen-nat)
+ non-member (|> gen-nat
+ (R;filter (. not (&;member? setL))))
+ #let [(^open "&/") &;Eq<Set>]]
+ (all (assert "I can query the size of a set."
+ (and (=+ sizeL (&;size setL))
+ (=+ sizeR (&;size setR))))
+
+ (assert "Converting sets to/from lists can't change their values."
+ (|> setL
+ &;to-list (&;from-list number;Hash<Nat>)
+ (&/= setL)))
+
+ (assert "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))))
+
+ (assert "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))))
+
+ (assert "Union with the empty set leaves a set unchanged."
+ (&/= setL
+ (&;union (&;new number;Hash<Nat>)
+ setL)))
+
+ (assert "Intersection with the empty set results in the empty set."
+ (let [empty-set (&;new number;Hash<Nat>)]
+ (&/= empty-set
+ (&;intersection empty-set setL))))
+
+ (assert "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? setL) (&;to-list setR)))))
+
+ (assert "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/test/test/lux/data/struct/stack.lux b/stdlib/test/test/lux/data/struct/stack.lux
new file mode 100644
index 000000000..dc3bb1e89
--- /dev/null
+++ b/stdlib/test/test/lux/data/struct/stack.lux
@@ -0,0 +1,47 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data (struct ["&" stack]
+ [list "" Fold<List>])
+ [number])
+ (math ["R" random])
+ pipe)
+ lux/test)
+
+(def: gen-nat
+ (R;Random Nat)
+ (|> R;nat
+ (:: R;Monad<Random> map (%+ +100))))
+
+(test: "Stacks"
+ [size gen-nat
+ sample (R;stack size gen-nat)
+ new-top gen-nat]
+ (all (assert "Can query the size of a stack."
+ (=+ size (&;size sample)))
+
+ (assert "Can peek inside non-empty stacks."
+ (case (&;peek sample)
+ #;None (&;empty? sample)
+ (#;Some _) (not (&;empty? sample))))
+
+ (assert "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 (=+ (&;size sample) (inc+ (&;size sample')))
+ (and (&;empty? sample) (&;empty? sample')))
+ ))
+
+ (assert "Pushing onto a stack always increases it by 1, adding a new value at the top."
+ (and (== sample
+ (&;pop (&;push new-top sample)))
+ (=+ (inc+ (&;size sample)) (&;size (&;push new-top sample)))
+ (|> (&;push new-top sample) &;peek (default (undefined))
+ (== new-top))))
+ ))
diff --git a/stdlib/test/test/lux/data/struct/tree.lux b/stdlib/test/test/lux/data/struct/tree.lux
new file mode 100644
index 000000000..0595ca7b3
--- /dev/null
+++ b/stdlib/test/test/lux/data/struct/tree.lux
@@ -0,0 +1,39 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data (struct ["&" tree]
+ [list "List/" Monad<List>])
+ [number])
+ (math ["R" random])
+ pipe)
+ lux/test)
+
+(def: gen-nat
+ (R;Random Nat)
+ (|> R;nat
+ (:: R;Monad<Random> map (%+ +100))))
+
+(test: "Trees"
+ [leaf (:: @ map &;leaf R;nat)
+ branchS gen-nat
+ branchV R;nat
+ branchC (R;list branchS R;nat)
+ #let [branch (&;branch branchV (List/map &;leaf branchC))]
+ #let [(^open "&/") (&;Eq<Tree> number;Eq<Nat>)
+ (^open "List/") (list;Eq<List> number;Eq<Nat>)]]
+ (all (assert "Can compare trees for equality."
+ (and (&/= leaf leaf)
+ (&/= branch branch)
+ (not (&/= leaf branch))
+ (not (&/= leaf (&;branch branchV (List/map &;leaf (list;reverse branchC)))))))
+
+ (assert "Can flatten a tree to get all the nodes as a flat tree."
+ (List/= (list& branchV branchC)
+ (&;flatten branch)))
+ ))
diff --git a/stdlib/test/test/lux/data/struct/vector.lux b/stdlib/test/test/lux/data/struct/vector.lux
new file mode 100644
index 000000000..87f8fa4cb
--- /dev/null
+++ b/stdlib/test/test/lux/data/struct/vector.lux
@@ -0,0 +1,84 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data (struct ["&" vector]
+ [list "List/" Fold<List> Functor<List>])
+ [text "Text/" Monoid<Text>]
+ text/format
+ [number])
+ (codata function)
+ (math ["R" random])
+ pipe)
+ lux/test)
+
+(test: "Vectors"
+ [size (|> R;nat (:: @ map (%+ +100)))
+ idx (|> R;nat (:: @ map (%+ size)))
+ sample (R;vector size R;nat)
+ other-sample (R;vector size R;nat)
+ non-member (|> R;nat (R;filter (. not (&;member? number;Eq<Nat> sample))))
+ #let [(^open "&/") (&;Eq<Vector> number;Eq<Nat>)
+ (^open "&/") &;Monad<Vector>
+ (^open "&/") &;Fold<Vector>
+ (^open "&/") &;Monoid<Vector>]]
+ (all (assert "Can query size of vector."
+ (if (&;empty? sample)
+ (and (=+ +0 size)
+ (=+ +0 (&;size sample)))
+ (=+ size (&;size sample))))
+
+ (assert "Can add and remove elements to vectors."
+ (and (=+ (inc+ size)
+ (&;size (&;add non-member sample)))
+ (=+ (dec+ size)
+ (&;size (&;pop sample)))))
+
+ (assert "Can put and get elements into vectors."
+ (|> sample
+ (&;put idx non-member)
+ (&;at idx)
+ (default (undefined))
+ (== non-member)))
+
+ (assert "Can update elements of vectors."
+ (|> sample
+ (&;put idx non-member)
+ (&;update idx inc+)
+ (&;at idx)
+ (default (undefined))
+ (=+ (inc+ non-member))))
+
+ (assert "Can safely transform to/from lists."
+ (|> sample
+ &;to-list &;from-list
+ (&/= sample)))
+
+ (assert "Can identify members of a vector."
+ (and (not (&;member? number;Eq<Nat> sample non-member))
+ (&;member? number;Eq<Nat> (&;add non-member sample) non-member)))
+
+ (assert "Can fold over elements of vector."
+ (=+ (List/fold ++ +0 (&;to-list sample))
+ (&/fold ++ +0 sample)))
+
+ (assert "Functor goes over every element."
+ (let [there (&/map inc+ sample)
+ back-again (&/map dec+ there)]
+ (and (not (&/= sample there))
+ (&/= sample back-again))))
+
+ (assert "Applicative allows you to create singleton vectors, and apply vectors of functions to vectors of values."
+ (and (&/= (&;vector non-member) (&/wrap non-member))
+ (&/= (&/map inc+ sample)
+ (&/apply (&/wrap inc+) sample))))
+
+ (assert "Vector concatenation is a monad."
+ (&/= (&/append sample other-sample)
+ (&/join (&;vector sample other-sample))))
+ ))
diff --git a/stdlib/test/test/lux/data/struct/zipper.lux b/stdlib/test/test/lux/data/struct/zipper.lux
new file mode 100644
index 000000000..a3bede88d
--- /dev/null
+++ b/stdlib/test/test/lux/data/struct/zipper.lux
@@ -0,0 +1,127 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data (struct ["&" zipper]
+ [tree]
+ [list "List/" Fold<List> Functor<List>])
+ [text "Text/" Monoid<Text>]
+ text/format
+ [number])
+ (codata function)
+ (math ["R" random])
+ pipe)
+ lux/test)
+
+(def: gen-tree
+ (R;Random (tree;Tree Nat))
+ (R;rec (lambda [gen-tree]
+ (do R;Monad<Random>
+ ## Each branch can have, at most, 1 child.
+ [size (|> R;nat (:: @ map (%+ +2)))]
+ (R;seq R;nat
+ (R;list size gen-tree))))))
+
+(def: (to-end zipper)
+ (All [a] (-> (&;Zipper a) (&;Zipper a)))
+ (loop [zipper zipper]
+ (if (&;end? zipper)
+ zipper
+ (recur (&;next zipper)))))
+
+(test: "Zippers"
+ [sample gen-tree
+ new-val R;nat
+ pre-val R;nat
+ post-val R;nat
+ #let [(^open "Tree/") (tree;Eq<Tree> number;Eq<Nat>)
+ (^open "List/") (list;Eq<List> number;Eq<Nat>)]]
+ (all (assert "Trees can be converted to/from zippers."
+ (|> sample
+ &;from-tree &;to-tree
+ (Tree/= sample)))
+
+ (assert "Creating a zipper gives you a root node."
+ (|> sample &;from-tree &;root?))
+
+ (assert "Can move down inside branches. Can move up from lower nodes."
+ (let [zipper (&;from-tree sample)]
+ (if (&;branch? zipper)
+ (let [child (|> zipper &;down)]
+ (and (not (Tree/= sample (&;to-tree child)))
+ (|> child &;parent (default (undefined)) (== zipper))
+ (|> child &;up (== zipper))
+ (|> child &;root (== zipper))))
+ (and (&;leaf? zipper)
+ (|> zipper (&;prepend-child new-val) &;branch?)))))
+
+ (assert "Can prepend and append children."
+ (let [zipper (&;from-tree sample)]
+ (if (&;branch? zipper)
+ (let [mid-val (|> zipper &;down &;value)
+ zipper (|> zipper
+ (&;prepend-child pre-val)
+ (&;append-child post-val))]
+ (and (|> zipper &;down &;value (== pre-val))
+ (|> zipper &;down &;right &;value (== mid-val))
+ (|> zipper &;down &;right &;right &;value (== post-val))
+ (|> zipper &;down &;rightmost &;leftmost &;value (== pre-val))
+ (|> zipper &;down &;right &;left &;value (== mid-val))
+ (|> zipper &;down &;rightmost &;value (== post-val))))
+ true)))
+
+ (assert "Can insert children around a node (unless it's root)."
+ (let [zipper (&;from-tree sample)]
+ (if (&;branch? zipper)
+ (let [mid-val (|> zipper &;down &;value)
+ zipper (|> zipper
+ &;down
+ (&;insert-left pre-val)
+ (default (undefined))
+ (&;insert-right post-val)
+ (default (undefined))
+ &;up)]
+ (and (|> zipper &;down &;value (== pre-val))
+ (|> zipper &;down &;right &;value (== mid-val))
+ (|> zipper &;down &;right &;right &;value (== post-val))
+ (|> zipper &;down &;rightmost &;leftmost &;value (== pre-val))
+ (|> zipper &;down &;right &;left &;value (== mid-val))
+ (|> zipper &;down &;rightmost &;value (== post-val))))
+ (and (|> zipper (&;insert-left pre-val) (case> (#;Some _) false
+ #;None true))
+ (|> zipper (&;insert-right post-val) (case> (#;Some _) false
+ #;None true))))))
+
+ (assert "Can set and update the value of a node."
+ (|> sample &;from-tree (&;set new-val) &;value (=+ new-val)))
+
+ (assert "Zipper traversal follows the outline of the tree depth-first."
+ (List/= (tree;flatten sample)
+ (loop [zipper (&;from-tree sample)]
+ (if (&;end? zipper)
+ (list)
+ (#;Cons (&;value zipper)
+ (recur (&;next zipper)))))))
+
+ (assert "Backwards zipper traversal yield reverse tree flatten."
+ (List/= (list;reverse (tree;flatten sample))
+ (loop [zipper (to-end (&;from-tree sample))]
+ (if (&;root? zipper)
+ (list)
+ (#;Cons (&;value zipper)
+ (recur (&;prev zipper)))))))
+
+ (assert "Can remove nodes (except root nodes)."
+ (let [zipper (&;from-tree 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/test/test/lux/data/sum.lux b/stdlib/test/test/lux/data/sum.lux
new file mode 100644
index 000000000..a23eeec00
--- /dev/null
+++ b/stdlib/test/test/lux/data/sum.lux
@@ -0,0 +1,32 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data sum
+ [text "Text/" Monoid<Text>]
+ [number])
+ (codata function))
+ lux/test)
+
+(test: "Sum operations"
+ (all (match (+0 1) (left 1))
+ (match (+1 2) (right 2))
+ (match (^ (list "0" "2"))
+ (lefts (: (List (| Text Text))
+ (list (+0 "0") (+1 "1") (+0 "2")))))
+ (match (^ (list "1"))
+ (rights (: (List (| Text Text))
+ (list (+0 "0") (+1 "1") (+0 "2")))))
+ (match (^ [(list "0" "2") (list "1")])
+ (partition (: (List (| Text Text))
+ (list (+0 "0") (+1 "1") (+0 "2")))))
+ (match 10
+ (either (lambda [_] 10) (lambda [_] 20) (: (| Text Text) (+0 ""))))
+ (match 20
+ (either (lambda [_] 10) (lambda [_] 20) (: (| Text Text) (+1 ""))))
+ ))
diff --git a/stdlib/test/test/lux/data/text.lux b/stdlib/test/test/lux/data/text.lux
new file mode 100644
index 000000000..640ae3f4c
--- /dev/null
+++ b/stdlib/test/test/lux/data/text.lux
@@ -0,0 +1,150 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data ["&" text]
+ [char]
+ text/format
+ [number]
+ (struct [list]))
+ (codata function)
+ (math ["R" random])
+ pipe)
+ lux/test)
+
+(test: "Size"
+ [size (:: @ map (%+ +100) R;nat)
+ sample (R;text size)]
+ (assert "" (or (and (=+ +0 size)
+ (&;empty? sample))
+ (=+ size (&;size sample)))))
+
+(def: bounded-size
+ (R;Random Nat)
+ (|> R;nat
+ (:: R;Monad<Random> map (|>. (%+ +100) (++ +1)))))
+
+(test: "Locations"
+ [size bounded-size
+ idx (:: @ map (%+ size) R;nat)
+ sample (R;text size)]
+ (assert "" (|> sample
+ (&;at idx)
+ (case> (^=> (#;Some char)
+ {(char;as-text char) char'}
+ {[(&;index-of char' sample)
+ (&;last-index-of char' sample)
+ (&;index-of' char' idx sample)
+ (&;last-index-of' char' idx sample)]
+ [(#;Some io) (#;Some lio)
+ (#;Some io') (#;Some lio')]})
+ (and (<=+ idx io)
+ (>=+ idx lio)
+
+ (=+ idx io')
+ (>=+ idx lio')
+
+ (&;contains? char' sample))
+
+ _
+ false
+ ))
+ ))
+
+(test: "Text functions"
+ [sizeL bounded-size
+ sizeR bounded-size
+ sampleL (R;text sizeL)
+ sampleR (R;text sizeR)
+ #let [sample (&;concat (list sampleL sampleR))
+ fake-sample (&;join-with " " (list sampleL sampleR))
+ dup-sample (&;join-with "" (list sampleL sampleR))
+ enclosed-sample (&;enclose [sampleR sampleR] sampleL)
+ (^open) &;Eq<Text>]]
+ (assert "" (and (not (= sample fake-sample))
+ (= sample dup-sample)
+ (&;starts-with? sampleL sample)
+ (&;ends-with? sampleR sample)
+ (= enclosed-sample
+ (&;enclose' sampleR sampleL))
+
+ (|> (&;split sizeL sample)
+ (case> (#;Right [_l _r])
+ (and (= sampleL _l)
+ (= sampleR _r)
+ (= sample (&;concat (list _l _r))))
+
+ _
+ false))
+
+ (|> [(&;sub +0 sizeL sample)
+ (&;sub sizeL (&;size sample) sample)
+ (&;sub' sizeL sample)
+ (&;sub' +0 sample)]
+ (case> [(#;Right _l) (#;Right _r) (#;Right _r') (#;Right _f)]
+ (and (= sampleL _l)
+ (= sampleR _r)
+ (= _r _r')
+ (= sample _f))
+
+ _
+ false))
+ )
+ ))
+
+(test: "More text functions"
+ [sizeS bounded-size
+ sizeP bounded-size
+ sizeL bounded-size
+ sep1 (R;text sizeS)
+ sep2 (R;text sizeS)
+ #let [part-gen (|> (R;text sizeP)
+ (R;filter (. not (&;contains? sep1))))]
+ parts (R;list sizeL part-gen)
+ #let [sample1 (&;concat (list;interpose sep1 parts))
+ sample2 (&;concat (list;interpose sep2 parts))
+ (^open) &;Eq<Text>]]
+ (assert "" (and (=+ (list;size parts)
+ (list;size (&;split-all-with sep1 sample1)))
+ (= sample2
+ (&;replace sep1 sep2 sample1))
+ )))
+
+(test: "Other text functions"
+ (all (match "abc" (&;lower-case "ABC"))
+ (match "ABC" (&;upper-case "abc"))
+ (match "ABC" (&;trim " \tABC\n\r"))
+ ))
+
+(test: "Structures"
+ (all (assert "" (:: &;Ord<Text> < "bcd" "abc"))
+ (assert "" (not (:: &;Ord<Text> < "abc" "abc")))
+ (assert "" (not (:: &;Ord<Text> < "abc" "bcd")))
+ (assert "" (:: &;Ord<Text> <= "bcd" "abc"))
+ (assert "" (:: &;Ord<Text> <= "abc" "abc"))
+ (assert "" (not (:: &;Ord<Text> <= "abc" "bcd")))
+ (assert "" (:: &;Ord<Text> > "abc" "bcd"))
+ (assert "" (not (:: &;Ord<Text> > "abc" "abc")))
+ (assert "" (not (:: &;Ord<Text> > "bcd" "abc")))
+ (assert "" (:: &;Ord<Text> >= "abc" "bcd"))
+ (assert "" (:: &;Ord<Text> >= "abc" "abc"))
+ (assert "" (not (:: &;Ord<Text> >= "bcd" "abc")))
+ ))
+
+(test: "Codec"
+ [size bounded-size
+ sample (R;text size)
+ #let [(^open) &;Eq<Text>]]
+ (assert "" (|> sample
+ (:: &;Codec<Text,Text> encode)
+ (:: &;Codec<Text,Text> decode)
+ (case> (#;Right decoded)
+ (= sample decoded)
+
+ _
+ false))))
diff --git a/stdlib/test/test/lux/data/text/format.lux b/stdlib/test/test/lux/data/text/format.lux
new file mode 100644
index 000000000..cd15c8584
--- /dev/null
+++ b/stdlib/test/test/lux/data/text/format.lux
@@ -0,0 +1,22 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data text/format
+ [number])
+ (codata function))
+ lux/test)
+
+(test: "Formatters"
+ (all (match "true" (%b true))
+ (match "123" (%i 123))
+ (match "123.456" (%r 123.456))
+ (match "#\"t\"" (%c #"t"))
+ (match "\"YOLO\"" (%t "YOLO"))
+ (match "User-id: 123 -- Active: true" (format "User-id: " (%i 123) " -- Active: " (%b true)))
+ ))
diff --git a/stdlib/test/test/lux/host.lux b/stdlib/test/test/lux/host.lux
new file mode 100644
index 000000000..109d8dfed
--- /dev/null
+++ b/stdlib/test/test/lux/host.lux
@@ -0,0 +1,54 @@
+(;module:
+ lux
+ (lux (control monad)
+ (data text/format
+ [number]
+ [product])
+ (codata function
+ [io])
+ host)
+ lux/test)
+
+(jvm-import java.lang.Object
+ (new []))
+
+(jvm-import java.lang.String)
+
+(jvm-import (java.lang.Class a)
+ (getName [] String))
+
+(test: "lux/host exports"
+ (let% [<conversions-0> (do-template [<value> <forward> <backward>]
+ [(match <value> (|> <value> <forward> <backward>))]
+
+ [123 l2d d2l]
+ [123 l2f f2l]
+ [123 l2i i2l]
+ [123.0 d2l l2d]
+ [123.0 d2f f2d]
+ [123.0 d2i i2d]
+ )
+ <conversions-1> (do-template [<forward> <backward>]
+ [(match 123 (|> 123 l2i <forward> <backward> i2l))]
+
+ [i2c c2i]
+ )]
+ (test-all (match "java.lang.Class" (Class.getName [] (class-for java.lang.Class)))
+ (match "java.lang.Class" (Class.getName [] (class-for Class)))
+ (match true (null? (: Object (null))))
+ (match false (null? (Object.new [])))
+ (match #;None (: (Maybe Object) (??? (null))))
+ (match (#;Some _) (: (Maybe Object) (??? (Object.new []))))
+ (match true (null? (!!! (: (Maybe Object) (??? (null))))))
+ (match false (null? (!!! (: (Maybe Object) (??? (Object.new []))))))
+ (match true (instance? Object (Object.new [])))
+ (match false (instance? String (Object.new [])))
+ (match 123 (synchronized (Object.new [])
+ 123))
+ (match +10 (array-length (array String +10)))
+ (match "YOLO" (let [array (array String +10)]
+ (exec (array-store +0 "YOLO" array)
+ (array-load +0 array))))
+ <conversions-0>
+ <conversions-1>
+ )))
diff --git a/stdlib/test/test/lux/lexer.lux b/stdlib/test/test/lux/lexer.lux
new file mode 100644
index 000000000..d0b17fe4b
--- /dev/null
+++ b/stdlib/test/test/lux/lexer.lux
@@ -0,0 +1,133 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ [lux #- not]
+ (lux (control monad)
+ (codata [io])
+ (data error)
+ [test #- fail assert]
+ lexer))
+
+## [Tests]
+(test: "Lexer end works"
+ (test-all (should-pass (run end ""))
+ (should-fail (run end "YOLO"))))
+
+(test: "Simple text lexers"
+ (test-all (match (#;Right "YO")
+ (run (this "YO") "YOLO"))
+ (should-fail (run (this "YO") "MEME"))))
+
+(test: "Char lexers"
+ (test-all (match (#;Right #"Y")
+ (run (this-char #"Y") "YOLO"))
+ (should-fail (run (this-char #"Y") "MEME"))
+ (match (#;Right #"Y")
+ (run (char-range #"X" #"Z") "YOLO"))
+ (should-fail (run (char-range #"X" #"Z") "MEME"))
+ (match (#;Right #"Y")
+ (run upper "YOLO"))
+ (should-fail (run upper "meme"))
+ (match (#;Right #"y")
+ (run lower "yolo"))
+ (should-fail (run lower "MEME"))
+ (match (#;Right #"1")
+ (run digit "1"))
+ (should-fail (run digit " "))
+ (match (#;Right #"7")
+ (run oct-digit "7"))
+ (should-fail (run oct-digit "8"))
+ (match (#;Right #"A")
+ (run any "A"))
+ (should-fail (run any ""))))
+
+(test: "Combinators"
+ (test-all (match (#;Right [#"Y" #"O"])
+ (run (seq any any) "YOLO"))
+ (should-fail (run (seq any any) "Y"))
+ (match+ (#;Left #"0")
+ (should-pass (run (alt digit upper) "0")))
+ (match+ (#;Right #"A")
+ (should-pass (run (alt digit upper) "A")))
+ (should-fail (run (alt digit upper) "a"))
+ (should-pass (run (not (alt digit upper)) "a"))
+ (should-fail (run (not (alt digit upper)) "A"))
+ (match (#;Right #"0")
+ (run (either digit upper) "0"))
+ (match (#;Right #"A")
+ (run (either digit upper) "A"))
+ (should-fail (run (either digit upper) "a"))
+ (match (#;Right #"A")
+ (run alpha "A"))
+ (match (#;Right #"a")
+ (run alpha "a"))
+ (should-fail (run alpha "1"))
+ (match (#;Right #"A")
+ (run alpha-num "A"))
+ (match (#;Right #"a")
+ (run alpha-num "a"))
+ (match (#;Right #"1")
+ (run alpha-num "1"))
+ (should-fail (run alpha-num " "))
+ (match (#;Right #"1")
+ (run hex-digit "1"))
+ (match (#;Right #"a")
+ (run hex-digit "a"))
+ (match (#;Right #"A")
+ (run hex-digit "A"))
+ (should-fail (run hex-digit " "))
+ (match (#;Right #" ")
+ (run space " "))
+ (should-fail (run space "8"))
+ (match (#;Right #"C")
+ (run (one-of "ABC") "C"))
+ (should-fail (run (one-of "ABC") "D"))
+ (match (#;Right #"D")
+ (run (none-of "ABC") "D"))
+ (should-fail (run (none-of "ABC") "C"))
+ (match (#;Right #"D")
+ (run (satisfies (lambda [c] true)) "D"))
+ (should-fail (run (satisfies (lambda [c] false)) "C"))
+ (match (#;Right "0123456789ABCDEF")
+ (run (many' hex-digit) "0123456789ABCDEF yolo"))
+ (should-fail (run (many' hex-digit) "yolo"))
+ (match (#;Right "")
+ (run (some' hex-digit) "yolo"))
+ ))
+
+(test: "Yet more combinators..."
+ (test-all (should-fail (run (fail "Well, it really SHOULD fail...") "yolo"))
+ (should-fail (run (assert false "Well, it really SHOULD fail...") "yolo"))
+ (should-pass (run (assert true "GO, GO, GO!") "yolo"))
+ (match (^ (#;Right (list #"0" #"1" #"2" #"3" #"4" #"5" #"6" #"7" #"8" #"9" #"A" #"B" #"C" #"D" #"E" #"F")))
+ (run (many hex-digit) "0123456789ABCDEF yolo"))
+ (should-fail (run (many hex-digit) "yolo"))
+ (match (^ (#;Right (list)))
+ (run (some hex-digit) "yolo"))
+ (match (^ (#;Right (list #"0" #"1" #"2" #"3" #"4" #"5" #"6" #"7" #"8" #"9" #"A" #"B" #"C" #"D" #"E" #"F")))
+ (run (exactly +16 hex-digit) "0123456789ABCDEF yolo"))
+ (match (^ (#;Right (list #"0" #"1" #"2")))
+ (run (exactly +3 hex-digit) "0123456789ABCDEF yolo"))
+ (should-fail (run (exactly +17 hex-digit) "0123456789ABCDEF yolo"))
+ (match (^ (#;Right (list #"0" #"1" #"2" #"3" #"4" #"5" #"6" #"7" #"8" #"9" #"A" #"B" #"C" #"D" #"E" #"F")))
+ (run (at-most +16 hex-digit) "0123456789ABCDEF yolo"))
+ (match (^ (#;Right (list #"0" #"1" #"2")))
+ (run (at-most +3 hex-digit) "0123456789ABCDEF yolo"))
+ (match (^ (#;Right (list #"0" #"1" #"2" #"3" #"4" #"5" #"6" #"7" #"8" #"9" #"A" #"B" #"C" #"D" #"E" #"F")))
+ (run (at-most +17 hex-digit) "0123456789ABCDEF yolo"))
+ (match (^ (#;Right (list #"0" #"1" #"2" #"3" #"4" #"5" #"6" #"7" #"8" #"9" #"A" #"B" #"C" #"D" #"E" #"F")))
+ (run (between +0 +16 hex-digit) "0123456789ABCDEF yolo"))
+ (match (^ (#;Right (list #"0" #"1" #"2" #"3" #"4" #"5" #"6" #"7" #"8" #"9" #"A" #"B" #"C" #"D" #"E" #"F")))
+ (run (between +3 +16 hex-digit) "0123456789ABCDEF yolo"))
+ (should-fail (run (between +17 +100 hex-digit) "0123456789ABCDEF yolo"))
+ (match (^ (#;Right (list #"0" #"1" #"2" #"3" #"4" #"5" #"6" #"7" #"8" #"9" #"A" #"B" #"C" #"D" #"E" #"F")))
+ (run (between +15 +20 hex-digit) "0123456789ABCDEF yolo"))
+ (match (#;Right (#;Some #"1")) (run (opt hex-digit) "123abc"))
+ (match (#;Right #;None) (run (opt hex-digit) "yolo"))
+ (match (^ (#;Right (list #"0" #"1" #"2" #"3" #"4" #"5" #"6" #"7" #"8" #"9" #"a" #"b" #"c" #"d" #"e" #"f")))
+ (run (sep-by space hex-digit) "0 1 2 3 4 5 6 7 8 9 a b c d e f YOLO"))
+ (match (#;Right "yolo") (run get-input "yolo"))
+ ))
diff --git a/stdlib/test/test/lux/macro/ast.lux b/stdlib/test/test/lux/macro/ast.lux
new file mode 100644
index 000000000..b06efce01
--- /dev/null
+++ b/stdlib/test/test/lux/macro/ast.lux
@@ -0,0 +1,31 @@
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data [text "Text/" Monoid<Text>]
+ [number])
+ (macro ast)
+ (codata function))
+ lux/test)
+
+(test: "lux/macro/ast exports"
+ (let% [<tests> (do-template [<expr> <text> <pattern>]
+ [(match <pattern> <expr>)
+ (match <text> (ast-to-text <expr>))
+ (match true (:: Eq<AST> = <expr> <expr>))]
+
+ [(bool true) "true" [["" -1 -1] (#;BoolS true)]]
+ [(bool false) "false" [_ (#;BoolS false)]]
+ [(int 123) "123" [_ (#;IntS 123)]]
+ [(real 123.0) "123.0" [_ (#;RealS 123.0)]]
+ [(char #"\n") "#\"\\n\"" [_ (#;CharS #"\n")]]
+ [(text "\n") "\"\\n\"" [_ (#;TextS "\n")]]
+ [(tag ["yolo" "lol"]) "#yolo;lol" [_ (#;TagS ["yolo" "lol"])]]
+ [(symbol ["yolo" "lol"]) "yolo;lol" [_ (#;SymbolS ["yolo" "lol"])]]
+ [(form (list (bool true) (int 123))) "(true 123)" (^ [_ (#;FormS (list [_ (#;BoolS true)] [_ (#;IntS 123)]))])]
+ [(tuple (list (bool true) (int 123))) "[true 123]" (^ [_ (#;TupleS (list [_ (#;BoolS true)] [_ (#;IntS 123)]))])]
+ [(record (list [(bool true) (int 123)])) "{true 123}" (^ [_ (#;RecordS (list [[_ (#;BoolS true)] [_ (#;IntS 123)]]))])]
+ [(local-tag "lol") "#lol" [_ (#;TagS ["" "lol"])]]
+ [(local-symbol "lol") "lol" [_ (#;SymbolS ["" "lol"])]]
+ )]
+ (test-all <tests>)))
diff --git a/stdlib/test/test/lux/macro/syntax.lux b/stdlib/test/test/lux/macro/syntax.lux
new file mode 100644
index 000000000..99f8550c0
--- /dev/null
+++ b/stdlib/test/test/lux/macro/syntax.lux
@@ -0,0 +1,176 @@
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data [text "Text/" Monoid<Text>]
+ [number])
+ (macro [ast]
+ ["s" syntax #+ syntax: Syntax])
+ (codata function))
+ lux/test)
+
+(test: "lux/macro/syntax exports [Part 1]"
+ (let% [<simple-tests> (do-template [<pattern> <expr> <get> <ask> <demand>]
+ [(match (#;Right [_ <pattern>])
+ (s;run (list <expr>)
+ <get>))
+ (match (#;Right [_ true])
+ (s;run (list <expr>)
+ (<ask> <pattern>)))
+ (match (#;Right [_ []])
+ (s;run (list <expr>)
+ (<demand> <pattern>)))]
+
+ [true (ast;bool true) s;bool s;bool? s;bool!]
+ [123 (ast;int 123) s;int s;int? s;int!]
+ [123.0 (ast;real 123.0) s;real s;real? s;real!]
+ [#"\n" (ast;char #"\n") s;char s;char? s;char!]
+ ["\n" (ast;text "\n") s;text s;text? s;text!]
+ [["yolo" "lol"] (ast;symbol ["yolo" "lol"]) s;symbol s;symbol? s;symbol!]
+ [["yolo" "lol"] (ast;tag ["yolo" "lol"]) s;tag s;tag? s;tag!]
+ )
+ <group-tests> (do-template [<parser> <ctor>]
+ [(match (#;Right [_ [true 123]])
+ (s;run (list (<ctor> (list (ast;bool true) (ast;int 123))))
+ (<parser> (s;seq s;bool s;int))))
+ (match (#;Right [_ true])
+ (s;run (list (<ctor> (list (ast;bool true))))
+ (<parser> s;bool)))
+ (match (#;Left _)
+ (s;run (list (<ctor> (list (ast;bool true) (ast;int 123))))
+ (<parser> s;bool)))
+ (match (#;Right [_ (#;Left true)])
+ (s;run (list (<ctor> (list (ast;bool true))))
+ (<parser> (s;alt s;bool s;int))))
+ (match (#;Right [_ (#;Right 123)])
+ (s;run (list (<ctor> (list (ast;int 123))))
+ (<parser> (s;alt s;bool s;int))))
+ (match (#;Left _)
+ (s;run (list (<ctor> (list (ast;real 123.0))))
+ (<parser> (s;alt s;bool s;int))))]
+
+ [s;form ast;form]
+ [s;tuple ast;tuple])]
+ (test-all (match (#;Right [_ [_ (#;BoolS true)]])
+ (s;run (list (ast;bool true) (ast;int 123))
+ s;any))
+ <simple-tests>
+ (match (#;Right [_ []])
+ (s;run (list (ast;bool true) (ast;int 123))
+ (s;assert true "yolo")))
+ (match (#;Left _)
+ (s;run (list (ast;bool true) (ast;int 123))
+ (s;assert false "yolo")))
+ (match (#;Right [_ +123])
+ (s;run (list (ast;nat +123))
+ s;nat))
+ (match (#;Left _)
+ (s;run (list (ast;int -123))
+ s;nat))
+ (match (#;Right [_ "yolo"])
+ (s;run (list (ast;local-symbol "yolo"))
+ s;local-symbol))
+ (match (#;Left _)
+ (s;run (list (ast;symbol ["yolo" "lol"]))
+ s;local-symbol))
+ (match (#;Right [_ "yolo"])
+ (s;run (list (ast;local-tag "yolo"))
+ s;local-tag))
+ (match (#;Left _)
+ (s;run (list (ast;tag ["yolo" "lol"]))
+ s;local-tag))
+ <group-tests>
+ )))
+
+(test: "lux/macro/syntax exports [Part 2]"
+ (test-all (match (#;Right [_ [true 123]])
+ (s;run (list (ast;record (list [(ast;bool true) (ast;int 123)])))
+ (s;record (s;seq s;bool s;int))))
+ (match (#;Right [_ (#;Some +123)])
+ (s;run (list (ast;nat +123))
+ (s;opt s;nat)))
+ (match (#;Right [_ #;None])
+ (s;run (list (ast;int -123))
+ (s;opt s;nat)))
+ (match (^ (#;Right [_ (list +123 +456 +789)]))
+ (s;run (list (ast;nat +123) (ast;nat +456) (ast;nat +789))
+ (s;some s;nat)))
+ (match (^ (#;Right [_ (list)]))
+ (s;run (list (ast;int -123))
+ (s;some s;nat)))
+ (match (^ (#;Right [_ (list +123 +456 +789)]))
+ (s;run (list (ast;nat +123) (ast;nat +456) (ast;nat +789))
+ (s;many s;nat)))
+ (match (^ (#;Right [_ (list +123)]))
+ (s;run (list (ast;nat +123))
+ (s;many s;nat)))
+ (match (#;Left _)
+ (s;run (list (ast;int -123))
+ (s;many s;nat)))
+ (match (#;Right [_ 123])
+ (s;run (list (ast;int 123) (ast;int 456) (ast;int 789))
+ (s;either s;pos-int s;int)))
+ (match (#;Right [_ -123])
+ (s;run (list (ast;int -123) (ast;int 456) (ast;int 789))
+ (s;either s;pos-int s;int)))
+ (match (#;Left _)
+ (s;run (list (ast;bool true) (ast;int 456) (ast;int 789))
+ (s;either s;pos-int s;int)))
+ (match (#;Right [_ true])
+ (s;run (list)
+ s;end?))
+ (match (#;Right [_ false])
+ (s;run (list (ast;bool true))
+ s;end?))
+ (match (#;Right [_ []])
+ (s;run (list)
+ s;end))
+ (match (#;Left _)
+ (s;run (list (ast;bool true))
+ s;end))
+ (match (^ (#;Right [_ (list 123 456 789)]))
+ (s;run (list (ast;int 123) (ast;int 456) (ast;int 789))
+ (s;exactly +3 s;int)))
+ (match (^ (#;Right [_ (list 123 456)]))
+ (s;run (list (ast;int 123) (ast;int 456) (ast;int 789))
+ (s;exactly +2 s;int)))
+ (match (#;Left _)
+ (s;run (list (ast;int 123) (ast;int 456) (ast;int 789))
+ (s;exactly +4 s;int)))
+ (match (^ (#;Right [_ (list 123 456 789)]))
+ (s;run (list (ast;int 123) (ast;int 456) (ast;int 789))
+ (s;at-least +3 s;int)))
+ (match (^ (#;Right [_ (list 123 456 789)]))
+ (s;run (list (ast;int 123) (ast;int 456) (ast;int 789))
+ (s;at-least +2 s;int)))
+ (match (#;Left _)
+ (s;run (list (ast;int 123) (ast;int 456) (ast;int 789))
+ (s;at-least +4 s;int)))
+ (match (^ (#;Right [_ (list 123 456 789)]))
+ (s;run (list (ast;int 123) (ast;int 456) (ast;int 789))
+ (s;at-most +3 s;int)))
+ (match (^ (#;Right [_ (list 123 456)]))
+ (s;run (list (ast;int 123) (ast;int 456) (ast;int 789))
+ (s;at-most +2 s;int)))
+ (match (^ (#;Right [_ (list 123 456 789)]))
+ (s;run (list (ast;int 123) (ast;int 456) (ast;int 789))
+ (s;at-most +4 s;int)))
+ (match (^ (#;Right [_ (list 123 456 789)]))
+ (s;run (list (ast;int 123) (ast;int 456) (ast;int 789))
+ (s;between +3 +10 s;int)))
+ (match (#;Left _)
+ (s;run (list (ast;int 123) (ast;int 456) (ast;int 789))
+ (s;between +4 +10 s;int)))
+ (match (^ (#;Right [_ (list 123 456 789)]))
+ (s;run (list (ast;int 123) (ast;text "YOLO") (ast;int 456) (ast;text "YOLO") (ast;int 789))
+ (s;sep-by (s;text! "YOLO") s;int)))
+ (match (^ (#;Right [_ (list 123 456)]))
+ (s;run (list (ast;int 123) (ast;text "YOLO") (ast;int 456) (ast;int 789))
+ (s;sep-by (s;text! "YOLO") s;int)))
+ (match (#;Left _)
+ (s;run (list (ast;int 123) (ast;int 456) (ast;int 789))
+ (s;not s;int)))
+ (match (#;Right [_ []])
+ (s;run (list (ast;bool true) (ast;int 456) (ast;int 789))
+ (s;not s;int)))
+ ))
diff --git a/stdlib/test/test/lux/math.lux b/stdlib/test/test/lux/math.lux
new file mode 100644
index 000000000..3d5e053f7
--- /dev/null
+++ b/stdlib/test/test/lux/math.lux
@@ -0,0 +1,45 @@
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data [text "Text/" Monoid<Text>]
+ text/format
+ [number]
+ (struct [list "List/" Fold<List> Functor<List>])
+ [product])
+ (codata function)
+ math)
+ lux/test)
+
+(test: "lux/math exports"
+ (test-all (match 1.0 (cos 0.0))
+ (match -1.0 (cos (/. 2.0 tau)))
+ ## (match 0.0 (cos (/. 4.0 tau)))
+ ## (match 0.0 (cos (*. (/. 4.0 3.0) tau)))
+
+ (match 1.0 (sin (/. 4.0 tau)))
+ (match -1.0 (sin (*. (/. 4.0 3.0) tau)))
+ ## (match 0.0 (sin 0.0))
+ ## (match 0.0 (sin (/. 2.0 tau)))
+
+ (match 4 (ceil 3.75))
+ (match 3 (floor 3.75))
+ (match 4 (round 3.75))
+ (match 3 (round 3.25))
+
+ (match 3.0 (cbrt 27.0))
+ (match 4.0 (sqrt 16.0))
+
+ (match 90.0 (degrees (/. 4.0 tau)))
+ (match true (=. tau (radians (degrees tau))))
+
+ (match 9 (gcd 450 27))
+ (match 40 (lcm 10 8))
+
+ (match 27 (infix 27))
+ (match 9 (infix [27 gcd 450]))
+ (match 9 (infix [(* 3 9) gcd 450]))
+ (match true (infix [#and 27 < 450 < 2000]))
+ (match true (infix [#and 27 < 450 > 200]))
+ (match true (infix [[27 < 450] and [200 < 2000]]))
+ ))
diff --git a/stdlib/test/test/lux/pipe.lux b/stdlib/test/test/lux/pipe.lux
new file mode 100644
index 000000000..a601bbf98
--- /dev/null
+++ b/stdlib/test/test/lux/pipe.lux
@@ -0,0 +1,47 @@
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data text/format
+ [number]
+ [product]
+ identity)
+ (codata function)
+ pipe)
+ lux/test)
+
+(test: "lux/pipe exports"
+ (test-all (match 1 (|> 20
+ (* 3)
+ (+ 4)
+ (_> 0 inc)))
+ (match 10 (|> 5
+ (@> [(+ @ @)])))
+ (match 15 (|> 5
+ (?> [even?] [(* 2)]
+ [odd?] [(* 3)]
+ [(_> -1)])))
+ (match 15 (|> 5
+ (?> [even?] [(* 2)]
+ [odd?] [(* 3)])))
+ (match 10 (|> 1
+ (!> [(< 10)]
+ [inc])))
+ (match 20 (|> 5
+ (%> Monad<Identity>
+ [(* 3)]
+ [(+ 4)]
+ [inc])))
+ (match "five" (|> 5
+ (case> 0 "zero"
+ 1 "one"
+ 2 "two"
+ 3 "three"
+ 4 "four"
+ 5 "five"
+ 6 "six"
+ 7 "seven"
+ 8 "eight"
+ 9 "nine"
+ _ "???")))
+ ))
diff --git a/stdlib/test/test/lux/regex.lux b/stdlib/test/test/lux/regex.lux
new file mode 100644
index 000000000..66355bdca
--- /dev/null
+++ b/stdlib/test/test/lux/regex.lux
@@ -0,0 +1,200 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data error
+ [product])
+ [compiler]
+ (macro [ast]
+ ["s" syntax #+ syntax:])
+ test
+ [lexer]
+ regex))
+
+(syntax: (should-regex {veredict (s;alt s;bool s;any)} {regex s;text} {input s;text})
+ (case veredict
+ (+0 ?)
+ (if ?
+ (wrap (list (` (match+ (~ (ast;text input))
+ (should-pass (lexer;run (regex (~ (ast;text regex)))
+ (~ (ast;text input))))))))
+ (wrap (list (` (should-fail (lexer;run (regex (~ (ast;text regex)))
+ (~ (ast;text input))))))))
+
+ (+1 result)
+ (wrap (list (` (match+ (~ result)
+ (should-pass (lexer;run (regex (~ (ast;text regex)))
+ (~ (ast;text input))))))))))
+
+## [Tests]
+(test: "Regular Expressions [Basics]"
+ (test-all (should-regex true "a" "a")
+ (should-regex false "a" ".")
+ (should-regex true "\\." ".")
+ (should-regex false "\\." "a")
+ ))
+
+(test: "Regular Expressions [System character classes]"
+ (test-all (should-regex true "." "a")
+
+ (should-regex true "\\d" "0")
+ (should-regex false "\\d" "m")
+ (should-regex true "\\D" "m")
+ (should-regex false "\\D" "0")
+
+ (should-regex true "\\s" " ")
+ (should-regex false "\\s" "m")
+ (should-regex true "\\S" "m")
+ (should-regex false "\\S" " ")
+
+ (should-regex true "\\w" "_")
+ (should-regex false "\\w" "^")
+ (should-regex true "\\W" ".")
+ (should-regex false "\\W" "a")
+
+ (should-regex true "\\p{Lower}" "m")
+ (should-regex false "\\p{Lower}" "M")
+
+ (should-regex true "\\p{Upper}" "M")
+ (should-regex false "\\p{Upper}" "m")
+
+ (should-regex true "\\p{Alpha}" "M")
+ (should-regex false "\\p{Alpha}" "0")
+
+ (should-regex true "\\p{Digit}" "1")
+ (should-regex false "\\p{Digit}" "n")
+
+ (should-regex true "\\p{Alnum}" "1")
+ (should-regex false "\\p{Alnum}" ".")
+
+ (should-regex true "\\p{Space}" " ")
+ (should-regex false "\\p{Space}" ".")
+
+ (should-regex true "\\p{HexDigit}" "a")
+ (should-regex false "\\p{HexDigit}" ".")
+
+ (should-regex true "\\p{OctDigit}" "6")
+ (should-regex false "\\p{OctDigit}" ".")
+
+ (should-regex true "\\p{Blank}" "\t")
+ (should-regex false "\\p{Blank}" ".")
+
+ (should-regex true "\\p{ASCII}" "\t")
+ (should-regex false "\\p{ASCII}" "\u1234")
+
+ (should-regex true "\\p{Contrl}" "\u0012")
+ (should-regex false "\\p{Contrl}" "a")
+
+ (should-regex true "\\p{Punct}" "@")
+ (should-regex false "\\p{Punct}" "a")
+
+ (should-regex true "\\p{Graph}" "@")
+ (should-regex false "\\p{Graph}" " ")
+
+ (should-regex true "\\p{Print}" "\u0020")
+ (should-regex false "\\p{Print}" "\u1234")
+ ))
+
+(test: "Regular Expressions [Custom character classes]"
+ (test-all (should-regex true "[abc]" "a")
+ (should-regex false "[abc]" "m")
+
+ (should-regex true "[a-z]" "a")
+ (should-regex true "[a-z]" "m")
+ (should-regex true "[a-z]" "z")
+
+ (should-regex true "[a-zA-Z]" "a")
+ (should-regex true "[a-zA-Z]" "m")
+ (should-regex true "[a-zA-Z]" "z")
+ (should-regex true "[a-zA-Z]" "A")
+ (should-regex true "[a-zA-Z]" "M")
+ (should-regex true "[a-zA-Z]" "Z")
+
+ (should-regex false "[^abc]" "a")
+ (should-regex true "[^abc]" "m")
+
+ (should-regex false "[^a-z]" "a")
+ (should-regex true "[^a-z]" "0")
+ (should-regex false "[^a-zA-Z]" "a")
+ (should-regex true "[^a-zA-Z]" "0")
+
+ (should-regex false "[a-z&&[def]]" "a")
+ (should-regex true "[a-z&&[def]]" "d")
+
+ (should-regex true "[a-z&&[^bc]]" "a")
+ (should-regex false "[a-z&&[^bc]]" "b")
+
+ (should-regex true "[a-z&&[^m-p]]" "a")
+ (should-regex false "[a-z&&[^m-p]]" "m")
+ (should-regex false "[a-z&&[^m-p]]" "p")
+ ))
+
+(test: "Regular Expressions [Reference]"
+ (test-all (let [number (regex "\\d+")]
+ (should-regex ["809-345-6789" "809" "345" "6789"] "(\\@<number>)-(\\@<number>)-(\\@<number>)" "809-345-6789"))
+ ))
+
+(test: "Regular Expressions [Quantifiers]"
+ (test-all (should-regex "aa" "aa" "aa")
+
+ (should-regex "a" "a?" "a")
+ (should-regex "" "a?" "")
+
+ (should-regex "aaa" "a*" "aaa")
+ (should-regex "" "a*" "")
+
+ (should-regex "aaa" "a+" "aaa")
+ (should-regex "a" "a+" "a")
+ (should-regex false "a+" "")
+
+ (should-regex "aa" "a{2}" "aa")
+ (should-regex "a" "a{1}" "aa")
+ (should-regex false "a{3}" "aa")
+
+ (should-regex "aa" "a{1,}" "aa")
+ (should-regex "aa" "a{2,}" "aa")
+ (should-regex false "a{3,}" "aa")
+
+ (should-regex "a" "a{,1}" "aa")
+ (should-regex "aa" "a{,2}" "aa")
+ (should-regex "aa" "a{,3}" "aa")
+
+ (should-regex "a" "a{1,2}" "a")
+ (should-regex "aa" "a{1,2}" "aa")
+ (should-regex "aa" "a{1,2}" "aaa")
+ ))
+
+(test: "Regular Expressions [Groups]"
+ (test-all (should-regex ["abc" "b"] "a(.)c" "abc")
+ (should-regex ["abbbbbc" "bbbbb"] "a(b+)c" "abbbbbc")
+ (should-regex ["809-345-6789" "809" "345" "6789"] "(\\d{3})-(\\d{3})-(\\d{4})" "809-345-6789")
+ (should-regex ["809-345-6789" "809" "6789"] "(\\d{3})-(?:\\d{3})-(\\d{4})" "809-345-6789")
+ (should-regex ["809-809-6789" "809" "6789"] "(\\d{3})-\\0-(\\d{4})" "809-809-6789")
+ (should-regex ["809-809-6789" "809" "6789"] "(?<code>\\d{3})-\\k<code>-(\\d{4})" "809-809-6789")
+ (should-regex ["809-809-6789-6789" "809" "6789"] "(?<code>\\d{3})-\\k<code>-(\\d{4})-\\0" "809-809-6789-6789")
+
+ (should-regex ["809-345-6789" "809" ["345-6789" "345" "6789"]] "(\\d{3})-((\\d{3})-(\\d{4}))" "809-345-6789")
+ ))
+
+(test: "Regular Expressions [Alternation]"
+ (test-all (should-regex ["a" (+0 [])] "a|b" "a")
+ (should-regex ["b" (+1 [])] "a|b" "b")
+ (should-regex false "a|b" "c")
+
+ (should-regex ["abc" (+0 "b")] "a(.)c|b(.)d" "abc")
+ (should-regex ["bcd" (+1 "c")] "a(.)c|b(.)d" "bcd")
+ (should-regex false "a(.)c|b(.)d" "cde")
+
+ (should-regex ["abc" (+0 ["b" "c"])] "a(.)(.)|b(.)(.)" "abc")
+ (should-regex ["bcd" (+1 ["c" "d"])] "a(.)(.)|b(.)(.)" "bcd")
+ (should-regex false "a(.)(.)|b(.)(.)" "cde")
+
+ (should-regex ["809-345-6789" (+0 ["809" "345-6789" "345" "6789"])]
+ "(\\d{3})-((\\d{3})-(\\d{4}))|b(.)d"
+ "809-345-6789")
+ ))
diff --git a/stdlib/test/test/lux/type.lux b/stdlib/test/test/lux/type.lux
new file mode 100644
index 000000000..8fa871e70
--- /dev/null
+++ b/stdlib/test/test/lux/type.lux
@@ -0,0 +1,41 @@
+(;module:
+ lux
+ (lux (codata [io])
+ (control monad)
+ (data [text "Text/" Monoid<Text>]
+ [number])
+ type
+ (codata function))
+ lux/test)
+
+(test: "lux/type exports"
+ (let% [<eq-tests> (do-template [<type>]
+ [(match true (:: Eq<Type> = <type> <type>))]
+
+ [(#;HostT "java.util.List" (list Int))]
+ [#;UnitT]
+ [#;VoidT]
+ [(#;VarT +123)]
+ [(#;ExT +123)]
+ [(#;BoundT +123)]
+ [(#;LambdaT Bool Int)]
+ [(#;AppT List Int)]
+ [(#;NamedT ["" "Int-List"] (#;AppT List Int))]
+ [(#;SumT Bool Int)]
+ [(#;ProdT Bool Int)]
+ [(#;UnivQ (list) (#;ProdT Bool (#;BoundT +1)))]
+ [(#;ExQ (list) (#;ProdT Bool (#;BoundT +1)))]
+ )]
+ (test-all <eq-tests>
+ (match (^=> (#;Some _type) (:: Eq<Type> = _type (#;ProdT Bool Int)))
+ (apply-type (type (Meta Bool)) Int))
+ (match #;None (apply-type Text Bool))
+ (match true
+ (:: Eq<Type> =
+ (#;NamedT ["" "a"]
+ (#;ProdT Bool Int))
+ (un-alias (#;NamedT ["" "c"]
+ (#;NamedT ["" "b"]
+ (#;NamedT ["" "a"]
+ (#;ProdT Bool Int)))))))
+ )))
diff --git a/stdlib/test/tests.lux b/stdlib/test/tests.lux
new file mode 100644
index 000000000..7b760c0f1
--- /dev/null
+++ b/stdlib/test/tests.lux
@@ -0,0 +1,84 @@
+## Copyright (c) Eduardo Julian. All rights reserved.
+## This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
+## If a copy of the MPL was not distributed with this file,
+## You can obtain one at http://mozilla.org/MPL/2.0/.
+
+(;module:
+ lux
+ (lux (control monad)
+ (codata [io])
+ (concurrency [promise])
+ [cli #+ program:]
+ [test])
+ (test lux
+ (lux (data [bit]
+ [bool]
+ [char]
+ [error]
+ [ident]
+ [identity]
+ [log]
+ [maybe]
+ [number]
+ [product]
+ [sum]
+ [text]
+ [text/format]
+ (struct [array]
+ [dict]
+ [list]
+ [queue]
+ [set]
+ [stack]
+ [tree]
+ [vector]
+ [zipper]
+ )
+ )
+ ## (codata ["_;" io]
+ ## [env]
+ ## [state]
+ ## (struct [stream]))
+ ## (macro [ast]
+ ## [syntax])
+ ## [type]
+ ## (concurrency ["_;" promise]
+ ## [frp]
+ ## [stm]
+ ## [actor]
+ ## )
+ ## [host]
+ ## ["_;" cli]
+ ## [math]
+ ## [pipe]
+ ## [lexer]
+ ## [regex]
+ ## (data (format [json]))
+ )
+ )
+ ## (lux ## (codata [cont])
+ ## ## (data (struct [stack]
+ ## ## [tree]
+ ## ## [zipper])
+ ## ## (error exception))
+ ## ## (concurrency [atom])
+ ## ## [macro]
+ ## ## (macro [template]
+ ## ## [poly]
+ ## ## (poly ["poly_;" eq]
+ ## ## ["poly_;" text-encoder]
+ ## ## ["poly_;" functor]))
+ ## ## (math [ratio]
+ ## ## [complex]
+ ## ## [random])
+ ## ## (type [check] [auto])
+ ## ## (control [effect])
+ ## ["_;" lexer]
+ ## ["_;" regex]
+ ## (data (format ["_;" json]))
+ ## )
+ )
+
+## [Program]
+(program: args
+ (test;run))