diff options
Diffstat (limited to 'stdlib')
-rw-r--r-- | stdlib/source/lux/control/transform.lux | 336 |
1 files changed, 161 insertions, 175 deletions
diff --git a/stdlib/source/lux/control/transform.lux b/stdlib/source/lux/control/transform.lux index 6376a9554..aa78736df 100644 --- a/stdlib/source/lux/control/transform.lux +++ b/stdlib/source/lux/control/transform.lux @@ -3,8 +3,7 @@ (lux (control [eq #+ Eq] [hash #+ Hash]) (data [maybe] - (coll [list] - [set #+ Set] + (coll [set #+ Set] [dict #+ Dict])) (math ["r" random #+ PRNG]))) @@ -19,115 +18,103 @@ (type: #export (Consumer cv cs) [cs (-> cv cs (Result cs))]) -(type: #export (Step v s) - (#Continue s (Maybe v)) +## TODO: Replace with Maybe once new-luxc becomes the standard compiler. +(type: #export (Step v t) + (#Continue t (Maybe v)) #Stop) -(type: #export (Transform pv cv ts) - [ts (-> pv ts (Step cv ts))]) +(type: #export (Transform pv cv) + (-> pv (Step cv (Transform pv cv)))) ## [Primitives] (def: #export (compose prev next) - (All [a b c ps ns] - (-> (Transform a b ps) (Transform b c ns) - (Transform a c [ps ns]))) - (let [[init|prev step|prev] prev - [init|next step|next] next] - [[init|prev init|next] - (function [input [state|prev state|next]] - (case (step|prev input state|prev) - (#Continue state|prev' (#.Some temp)) - (case (step|next temp state|next) - (#Continue state|next' ?output) - (#Continue [state|prev' state|next'] ?output) - - #Stop - #Stop) - - (#Continue state|prev' #.None) - (#Continue [state|prev' state|next] #.None) - - #Stop - #Stop))])) + (All [a b c] + (-> (Transform a b) (Transform b c) (Transform a c))) + (function [input] + (case (prev input) + (#Continue prev' (#.Some temp)) + (case (next temp) + (#Continue next' ?output) + (#Continue (compose prev' next') ?output) + + #Stop + #Stop) + + (#Continue prev' #.None) + (#Continue (compose prev' next) #.None) + + #Stop + #Stop))) (def: #export (each left right) - (All [a l r ls rs] - (-> (Transform a l ls) (Transform a r rs) - (Transform a (& l r) [ls rs]))) - (let [[init|left step|left] left - [init|right step|right] right] - [[init|left init|right] - (function [input [state|left state|right]] - (case [(step|left input state|left) - (step|right input state|right)] - [(#Continue state|left' (#.Some output|left)) - (#Continue state|right' (#.Some output|right))] - (#Continue [state|left' state|right'] - (#.Some [output|left output|right])) - - (^or [#Stop _] [_ #Stop]) - #Stop - - [(#Continue state|left' #.None) _] - (#Continue [state|left' state|right] #.None) - - [_ (#Continue state|right' #.None)] - (#Continue [state|left state|right'] #.None)))])) + (All [a l r] + (-> (Transform a l) (Transform a r) (Transform a [l r]))) + (function [input] + (case [(left input) + (right input)] + [(#Continue left' ?output|left) + (#Continue right' ?output|right)] + (case [?output|left ?output|right] + [(#.Some output|left) (#.Some output|right)] + (#Continue (each left' right') + (#.Some [output|left output|right])) + + _ + (#Continue (each left' right') + #.None)) + + _ + #Stop))) (def: #export (either left right) - (All [a b ls rs] - (-> (Transform a b ls) (Transform a b rs) - (Transform a b [ls rs]))) - (let [[init|left step|left] left - [init|right step|right] right] - [[init|left init|right] - (function [input [state|left state|right]] - (case (step|left input state|left) - (#Continue state|left' (#.Some output)) - (#Continue [state|left' state|right] (#.Some output)) - - (^template [<case> <left-state> <done>] - <case> - (case (step|right input state|right) - (#Continue state|right' (#.Some output)) - (#Continue [<left-state> state|right'] (#.Some output)) - - (#Continue state|right' #.None) - (#Continue [<left-state> state|right'] #.None) - - #Stop - <done>)) - ([(#Continue state|left' #.None) state|left' (#Continue [state|left' state|right] #.None)] - [#Stop state|left #Stop]) - ))])) + (All [a b] + (-> (Transform a b) (Transform a b) (Transform a b))) + (function [input] + (case (left input) + (#Continue left' (#.Some output)) + (#Continue (either left' right) (#.Some output)) + + (^template [<case> <left-state> <done>] + <case> + (case (right input) + (#Continue right' (#.Some output)) + (#Continue (either <left-state> right') (#.Some output)) + + (#Continue right' #.None) + (#Continue (either <left-state> right') #.None) + + #Stop + <done>)) + ([(#Continue left' #.None) left' (#Continue (either left' right) #.None)] + [#Stop left #Stop]) + ))) (def: #export (run transform producer consumer) - (All [ts pv ps cv cs] - (-> (Transform pv cv ts) + (All [pv ps cv cs] + (-> (Transform pv cv) (Producer pv ps) (Consumer cv cs) cs)) - (let [[init|transform step] transform - [init|producer produce] producer + (let [[init|producer produce] producer [init|consumer consume] consumer] - (loop [state|transform init|transform + (loop [transform transform state|producer init|producer state|consumer init|consumer] ## TODO: Delete 'output' let-binding once new-luxc is the ## standard compiler. (let [output (case (produce state|producer) (#.Some [production state|producer']) - (case (step production state|transform) - (#Continue state|transform' (#.Some temp)) + (case (transform production) + (#Continue transform' (#.Some temp)) (case (consume temp state|consumer) (#Partial state|consumer') - (recur state|transform' state|producer' state|consumer') + (recur transform' state|producer' state|consumer') (#Total output) output) - (#Continue state|transform' #.None) - (recur state|transform' state|producer' state|consumer) + (#Continue transform' #.None) + (recur transform' state|producer' state|consumer) #Stop state|consumer) @@ -157,122 +144,121 @@ ## [Transforms] (def: #export (map f) - (All [a b] (-> (-> a b) (Transform a b Unit))) - [[] - (function [input state] - (#Continue state (#.Some (f input))))]) + (All [a b] (-> (-> a b) (Transform a b))) + (function self [input] + (#Continue self (#.Some (f input))))) (def: #export (map-indexed f) - (All [a b] (-> (-> Nat a b) (Transform a b Nat))) - [+0 - (function [input index] - (#Continue (n/inc index) (#.Some (f index input))))]) + (All [a b] (-> (-> Nat a b) (Transform a b))) + (loop [index +0] + (function [input] + (#Continue (recur (n/inc index)) (#.Some (f index input)))))) (def: #export (filter pred) - (All [a] (-> (-> a Bool) (Transform a a Unit))) - [[] - (function [input state] - (#Continue state (if (pred input) - (#.Some input) - #.None)))]) + (All [a] (-> (-> a Bool) (Transform a a))) + (function self [input] + (#Continue self (if (pred input) + (#.Some input) + #.None)))) (def: #export (keep f) - (All [a b] (-> (-> a (Maybe b)) (Transform a b Unit))) - [[] - (function [input state] - (#Continue state (f input)))]) + (All [a b] (-> (-> a (Maybe b)) (Transform a b))) + (function self [input] + (#Continue self (f input)))) (def: #export (keep-indexed f) - (All [a b] (-> (-> Nat a (Maybe b)) (Transform a b Nat))) - [+0 - (function [input index] - (#Continue (n/inc index) (f index input)))]) + (All [a b] (-> (-> Nat a (Maybe b)) (Transform a b))) + (loop [index +0] + (function [input] + (#Continue (recur (n/inc index)) (f index input))))) (def: #export (take amount) - (All [a] (-> Nat (Transform a a Nat))) - [amount - (function [input remaining] - (if (n/= +0 remaining) - #Stop - (#Continue (n/dec remaining) (#.Some input))))]) + (All [a] (-> Nat (Transform a a))) + (loop [remaining amount] + (function [input] + (if (n/= +0 remaining) + #Stop + (#Continue (recur (n/dec remaining)) (#.Some input)))))) (def: #export (drop amount) - (All [a] (-> Nat (Transform a a Nat))) - [amount - (function [input remaining] - (if (n/= +0 remaining) - (#Continue remaining (#.Some input)) - (#Continue (n/dec remaining) #.None)))]) + (All [a] (-> Nat (Transform a a))) + (loop [remaining amount] + (function [input] + (if (n/= +0 remaining) + (#Continue (recur remaining) (#.Some input)) + (#Continue (recur (n/dec remaining)) #.None))))) (def: #export (take-while pred) - (All [a] (-> (-> a Bool) (Transform a a Unit))) - [[] - (function [input state] - (if (pred input) - (#Continue state (#.Some input)) - #Stop))]) + (All [a] (-> (-> a Bool) (Transform a a))) + (function self [input] + (if (pred input) + (#Continue self (#.Some input)) + #Stop))) (def: #export (drop-while pred) - (All [a] (-> (-> a Bool) (Transform a a Bool))) - [true - (function [input dropping?] - (if (and dropping? - (pred input)) - (#Continue true #.None) - (#Continue false (#.Some input))))]) + (All [a] (-> (-> a Bool) (Transform a a))) + (loop [dropping? true] + (function [input] + (if (and dropping? + (pred input)) + (#Continue (recur true) #.None) + (#Continue (recur false) (#.Some input)))))) (def: #export (take-nth nth) - (All [a] (-> Nat (Transform a a Nat))) - [+0 - (function [input seen] - (let [mod (n/% nth (n/inc seen))] - (if (n/= +0 mod) - (#Continue mod (#.Some input)) - (#Continue mod #.None))))]) + (All [a] (-> Nat (Transform a a))) + (loop [seen +0] + (function [input] + (let [mod (n/% nth (n/inc seen))] + (if (n/= +0 mod) + (#Continue (recur mod) (#.Some input)) + (#Continue (recur mod) #.None)))))) (def: #export (drop-nth nth) - (All [a] (-> Nat (Transform a a Nat))) - [+0 - (function [input seen] - (let [mod (n/% nth (n/inc seen))] - (if (n/= +0 mod) - (#Continue mod #.None) - (#Continue mod (#.Some input)))))]) + (All [a] (-> Nat (Transform a a))) + (loop [seen +0] + (function [input] + (let [mod (n/% nth (n/inc seen))] + (if (n/= +0 mod) + (#Continue (recur mod) #.None) + (#Continue (recur mod) (#.Some input))))))) (def: #export (distinct Hash<a>) - (All [a] (-> (Hash a) (Transform a a (Set a)))) - [(set.new Hash<a>) - (function [input seen] - (if (set.member? seen input) - (#Continue seen #.None) - (#Continue (set.add input seen) (#.Some input))))]) - + (All [a] (-> (Hash a) (Transform a a))) + (loop [seen (set.new Hash<a>)] + (function [input] + (if (set.member? seen input) + (#Continue (recur seen) #.None) + (#Continue (recur (set.add input seen)) (#.Some input)))))) + +## TODO: Remove whenever feasible. +(def: helper|de-duplicate + (All [a] (-> (Eq a) (Maybe a))) + (function [_] #.None)) (def: #export (de-duplicate Eq<a>) - (All [a] (-> (Eq a) (Transform a a (Maybe a)))) - [#.None - (function [input last] - (case last - (^multi (#.Some last') (:: Eq<a> = last' input)) - (#Continue last #.None) + (All [a] (-> (Eq a) (Transform a a))) + (loop [last (helper|de-duplicate Eq<a>)] + (function [input] + (case last + (^multi (#.Some last') (:: Eq<a> = last' input)) + (#Continue (recur last) #.None) - _ - (#Continue (#.Some input) (#.Some input))))]) + _ + (#Continue (recur (#.Some input)) (#.Some input)))))) (def: #export (random probability prng) - (All [a] (-> Deg PRNG (Transform a a PRNG))) - [prng - (function [input prng] - (let [[prng' chance] (r.run prng r.deg)] - (if (d/< probability chance) - (#Continue prng' (#.Some input)) - (#Continue prng' #.None))))]) + (All [a] (-> Deg PRNG (Transform a a))) + (loop [prng prng] + (function [input] + (let [[prng' chance] (r.run prng r.deg)] + (if (d/< probability chance) + (#Continue (recur prng') (#.Some input)) + (#Continue (recur prng') #.None)))))) (def: #export (replace dict) - (All [a] (-> (Dict a a) (Transform a a Unit))) - [[] - (function [input state] - (|> dict - (dict.get input) - (maybe.default input) - #.Some - (#Continue state)))]) + (All [a] (-> (Dict a a) (Transform a a))) + (function self [input] + (|> dict + (dict.get input) + (maybe.default input) + #.Some + (#Continue self)))) |