From 367c56f33d72621120bcf00953f5fafffb028e97 Mon Sep 17 00:00:00 2001 From: Eduardo Julian Date: Mon, 3 Apr 2017 19:19:09 -0400 Subject: - Renamed lux/codata/* to lux/function/* and moved the lux/codata/coll/stream to lux/data/coll/stream. --- stdlib/source/lux/codata/coll/stream.lux | 142 ------------------------------ stdlib/source/lux/codata/cont.lux | 69 --------------- stdlib/source/lux/codata/function.lux | 20 ----- stdlib/source/lux/codata/reader.lux | 63 ------------- stdlib/source/lux/codata/state.lux | 125 -------------------------- stdlib/source/lux/codata/thunk.lux | 33 ------- stdlib/source/lux/concurrency/actor.lux | 2 +- stdlib/source/lux/concurrency/frp.lux | 1 - stdlib/source/lux/concurrency/promise.lux | 2 +- stdlib/source/lux/control/order.lux | 4 +- stdlib/source/lux/data/bool.lux | 3 +- stdlib/source/lux/data/coll/list.lux | 3 +- stdlib/source/lux/data/coll/set.lux | 3 +- stdlib/source/lux/data/coll/stream.lux | 142 ++++++++++++++++++++++++++++++ stdlib/source/lux/data/format/json.lux | 1 - stdlib/source/lux/function.lux | 20 +++++ stdlib/source/lux/function/cont.lux | 69 +++++++++++++++ stdlib/source/lux/function/reader.lux | 63 +++++++++++++ stdlib/source/lux/function/state.lux | 125 ++++++++++++++++++++++++++ stdlib/source/lux/function/thunk.lux | 33 +++++++ stdlib/source/lux/host.jvm.lux | 1 - 21 files changed, 459 insertions(+), 465 deletions(-) delete mode 100644 stdlib/source/lux/codata/coll/stream.lux delete mode 100644 stdlib/source/lux/codata/cont.lux delete mode 100644 stdlib/source/lux/codata/function.lux delete mode 100644 stdlib/source/lux/codata/reader.lux delete mode 100644 stdlib/source/lux/codata/state.lux delete mode 100644 stdlib/source/lux/codata/thunk.lux create mode 100644 stdlib/source/lux/data/coll/stream.lux create mode 100644 stdlib/source/lux/function.lux create mode 100644 stdlib/source/lux/function/cont.lux create mode 100644 stdlib/source/lux/function/reader.lux create mode 100644 stdlib/source/lux/function/state.lux create mode 100644 stdlib/source/lux/function/thunk.lux (limited to 'stdlib/source') diff --git a/stdlib/source/lux/codata/coll/stream.lux b/stdlib/source/lux/codata/coll/stream.lux deleted file mode 100644 index 3089fe1b2..000000000 --- a/stdlib/source/lux/codata/coll/stream.lux +++ /dev/null @@ -1,142 +0,0 @@ -(;module: - lux - (lux (control functor - monad - comonad) - [compiler #+ with-gensyms] - (macro ["s" syntax #+ syntax: Syntax]) - (data (coll [list "List/" Monad]) - bool) - (codata [cont #+ @lazy Cont]))) - -## [Types] -(type: #export (Stream a) - {#;doc "An infinite stream of lazily-evaluated values."} - (Cont [a (Stream a)])) - -## [Utils] -(def: (cycle' x xs init full) - (All [a] - (-> a (List a) a (List a) (Stream a))) - (case xs - #;Nil (@lazy [x (cycle' init full init full)]) - (#;Cons x' xs') (@lazy [x (cycle' x' xs' init full)]))) - -## [Functions] -(def: #export (iterate f x) - {#;doc "Create a stream by applying a function to a value, and to its result, on and on..."} - (All [a] - (-> (-> a a) a (Stream a))) - (@lazy [x (iterate f (f x))])) - -(def: #export (repeat x) - {#;doc "Repeat a value forever."} - (All [a] - (-> a (Stream a))) - (@lazy [x (repeat x)])) - -(def: #export (cycle xs) - {#;doc "Go over the elements of a list forever. - - The list shouldn't be empty."} - (All [a] - (-> (List a) (Maybe (Stream a)))) - (case xs - #;Nil #;None - (#;Cons x xs') (#;Some (cycle' x xs' x xs')))) - -(do-template [ ] - [(def: #export ( s) - (All [a] (-> (Stream a) )) - (let [[h t] (cont;run s)] - ))] - - [head a h] - [tail (Stream a) t]) - -(def: #export (nth idx s) - (All [a] (-> Nat (Stream a) a)) - (let [[h t] (cont;run s)] - (if (n.> +0 idx) - (nth (n.dec idx) t) - h))) - -(do-template [ ] - [(def: #export ( pred xs) - (All [a] - (-> (Stream a) (List a))) - (let [[x xs'] (cont;run xs)] - (if - (list& x ( xs')) - (list)))) - - (def: #export ( pred xs) - (All [a] - (-> (Stream a) (Stream a))) - (let [[x xs'] (cont;run xs)] - (if - ( xs') - xs))) - - (def: #export ( pred xs) - (All [a] - (-> (Stream a) [(List a) (Stream a)])) - (let [[x xs'] (cont;run xs)] - (if - (let [[tail next] ( xs')] - [(#;Cons [x tail]) next]) - [(list) xs])))] - - [take-while drop-while split-while (-> a Bool) (pred x) pred] - [take drop split Nat (n.> +0 pred) (n.dec pred)] - ) - -(def: #export (unfold step init) - {#;doc "A stateful way of infinitely calculating the values of a stream."} - (All [a b] - (-> (-> a [a b]) a (Stream b))) - (let [[next x] (step init)] - (@lazy [x (unfold step next)]))) - -(def: #export (filter p xs) - (All [a] (-> (-> a Bool) (Stream a) (Stream a))) - (let [[x xs'] (cont;run xs)] - (if (p x) - (@lazy [x (filter p xs')]) - (filter p xs')))) - -(def: #export (partition p xs) - {#;doc "Split a stream in two based on a predicate. - - The left side contains all entries for which the predicate is true. - - The right side contains all entries for which the predicate is false."} - (All [a] (-> (-> a Bool) (Stream a) [(Stream a) (Stream a)])) - [(filter p xs) (filter (complement p) xs)]) - -## [Structures] -(struct: #export _ (Functor Stream) - (def: (map f fa) - (let [[h t] (cont;run fa)] - (@lazy [(f h) (map f t)])))) - -(struct: #export _ (CoMonad Stream) - (def: functor Functor) - (def: unwrap head) - (def: (split wa) - (let [[head tail] (cont;run wa)] - (@lazy [wa (split tail)])))) - -## [Pattern-matching] -(syntax: #export (^stream& [patterns (s;form (s;many s;any))] body [branches (s;some s;any)]) - {#;doc (doc "Allows destructuring of streams in pattern-matching expressions." - "Caveat emptor: Only use it for destructuring, and not for testing values within the streams." - (let [(^stream& x y z _tail) (some-stream-func 1 2 3)] - (func x y z)))} - (with-gensyms [g!s] - (let [body+ (` (let [(~@ (List/join (List/map (lambda [pattern] - (list (` [(~ pattern) (~ g!s)]) - (` (cont;run (~ g!s))))) - patterns)))] - (~ body)))] - (wrap (list& g!s body+ branches))))) diff --git a/stdlib/source/lux/codata/cont.lux b/stdlib/source/lux/codata/cont.lux deleted file mode 100644 index 7f1b918f0..000000000 --- a/stdlib/source/lux/codata/cont.lux +++ /dev/null @@ -1,69 +0,0 @@ -(;module: - lux - (lux (macro [ast]) - (control functor - applicative - monad) - (data (coll list))) - (.. function)) - -## [Types] -(type: #export (Cont a) - {#;doc "Delimited continuations."} - (All [b] - (-> (-> a b) b))) - -## [Syntax] -(macro: #export (@lazy tokens state) - {#;doc (doc "Delays the evaluation of an expression, by wrapping it in a continuation 'thunk'." - (@lazy (some-computation some-input)))} - (case tokens - (^ (list value)) - (let [blank (ast;symbol ["" ""])] - (#;Right [state (list (` (;lambda [(~ blank)] ((~ blank) (~ value)))))])) - - _ - (#;Left "Wrong syntax for @lazy"))) - -## [Functions] -(def: #export (call/cc f) - {#;doc "Call with current continuation."} - (All [a b c] - (-> (-> (-> a (Cont b c)) - (Cont a c)) - (Cont a c))) - (lambda [k] - (f (lambda [a _] - (k a)) - k))) - -(def: #export (continue f thunk) - {#;doc "Forces a continuation thunk to be evaluated."} - (All [i o] - (-> (-> i o) (Cont i o) o)) - (thunk f)) - -(def: #export (run thunk) - {#;doc "Forces a continuation thunk to be evaluated."} - (All [a] - (-> (Cont a) a)) - (continue id thunk)) - -## [Structs] -(struct: #export _ (Functor Cont) - (def: (map f ma) - (lambda [k] (ma (. k f))))) - -(struct: #export _ (Applicative Cont) - (def: functor Functor) - - (def: (wrap a) - (@lazy a)) - - (def: (apply ff fa) - (@lazy ((run ff) (run fa))))) - -(struct: #export _ (Monad Cont) - (def: applicative Applicative) - - (def: join run)) diff --git a/stdlib/source/lux/codata/function.lux b/stdlib/source/lux/codata/function.lux deleted file mode 100644 index cddf5d472..000000000 --- a/stdlib/source/lux/codata/function.lux +++ /dev/null @@ -1,20 +0,0 @@ -(;module: - lux - (lux (control monoid))) - -## [Functions] -(def: #export (const c) - {#;doc "Create constant functions."} - (All [a b] (-> a (-> b a))) - (lambda [_] c)) - -(def: #export (flip f) - {#;doc "Flips the order of the arguments of a function."} - (All [a b c] - (-> (-> a b c) (-> b a c))) - (lambda [x y] (f y x))) - -## [Structures] -(struct: #export Monoid (Monoid (All [a] (-> a a))) - (def: unit id) - (def: append .)) diff --git a/stdlib/source/lux/codata/reader.lux b/stdlib/source/lux/codata/reader.lux deleted file mode 100644 index 955b4bba3..000000000 --- a/stdlib/source/lux/codata/reader.lux +++ /dev/null @@ -1,63 +0,0 @@ -(;module: - lux - (lux (control functor - applicative - ["M" monad #*]))) - -## [Types] -(type: #export (Reader r a) - {#;doc "Computations that have access to some environmental value."} - (-> r a)) - -## [Structures] -(struct: #export Functor (All [r] (Functor (Reader r))) - (def: (map f fa) - (lambda [env] - (f (fa env))))) - -(struct: #export Applicative (All [r] (Applicative (Reader r))) - (def: functor Functor) - - (def: (wrap x) - (lambda [env] x)) - - (def: (apply ff fa) - (lambda [env] - ((ff env) (fa env))))) - -(struct: #export Monad (All [r] (Monad (Reader r))) - (def: applicative Applicative) - - (def: (join mma) - (lambda [env] - (mma env env)))) - -## [Values] -(def: #export ask - {#;doc "Get the environment."} - (All [r] (Reader r r)) - (lambda [env] env)) - -(def: #export (local change reader-proc) - {#;doc "Run computation with a locally-modified environment."} - (All [r a] (-> (-> r r) (Reader r a) (Reader r a))) - (|>. change reader-proc)) - -(def: #export (run env reader-proc) - (All [r a] (-> r (Reader r a) a)) - (reader-proc env)) - -(struct: #export (ReaderT Monad) - {#;doc "Monad transformer for Reader."} - (All [M] (-> (Monad M) (All [e] (Monad (All [a] (Reader e (M a))))))) - (def: applicative (compA Applicative (get@ #M;applicative Monad))) - (def: (join eMeMa) - (lambda [env] - (do Monad - [eMa (run env eMeMa)] - (run env eMa))))) - -(def: #export lift-reader - {#;doc "Lift monadic values to the Reader wrapper."} - (All [M e a] (-> (M a) (Reader e (M a)))) - (:: Monad wrap)) diff --git a/stdlib/source/lux/codata/state.lux b/stdlib/source/lux/codata/state.lux deleted file mode 100644 index 5f5b96e4b..000000000 --- a/stdlib/source/lux/codata/state.lux +++ /dev/null @@ -1,125 +0,0 @@ -(;module: - lux - (lux (control functor - ["A" applicative #*] - ["M" monad #*]))) - -## [Types] -(type: #export (State s a) - {#;doc "Stateful computations."} - (-> s [s a])) - -## [Structures] -(struct: #export Functor (All [s] (Functor (State s))) - (def: (map f ma) - (lambda [state] - (let [[state' a] (ma state)] - [state' (f a)])))) - -(struct: #export Applicative (All [s] (Applicative (State s))) - (def: functor Functor) - - (def: (wrap a) - (lambda [state] - [state a])) - - (def: (apply ff fa) - (lambda [state] - (let [[state' f] (ff state) - [state'' a] (fa state')] - [state'' (f a)])))) - -(struct: #export Monad (All [s] (Monad (State s))) - (def: applicative Applicative) - - (def: (join mma) - (lambda [state] - (let [[state' ma] (mma state)] - (ma state'))))) - -## [Values] -(def: #export get - {#;doc "Read the current state."} - (All [s] (State s s)) - (lambda [state] - [state state])) - -(def: #export (put new-state) - {#;doc "Set the new state."} - (All [s] (-> s (State s Unit))) - (lambda [state] - [new-state []])) - -(def: #export (update change) - {#;doc "Compute the new state."} - (All [s] (-> (-> s s) (State s Unit))) - (lambda [state] - [(change state) []])) - -(def: #export (use user) - {#;doc "Run function on current state."} - (All [s a] (-> (-> s a) (State s a))) - (lambda [state] - [state (user state)])) - -(def: #export (local change action) - {#;doc "Run computation with a locally-modified state."} - (All [s a] (-> (-> s s) (State s a) (State s a))) - (lambda [state] - (let [[state' output] (action (change state))] - [state output]))) - -(def: #export (run state action) - {#;doc "Run a stateful computation."} - (All [s a] (-> s (State s a) [s a])) - (action state)) - -(struct: (Functor Functor) - (All [M s] (-> (Functor M) (Functor (All [a] (-> s (M [s a])))))) - (def: (map f sfa) - (lambda [state] - (:: Functor map (lambda [[s a]] [s (f a)]) - (sfa state))))) - -(struct: (Applicative Monad) - (All [M s] (-> (Monad M) (Applicative (All [a] (-> s (M [s a])))))) - (def: functor (Functor (get@ [#M;applicative #A;functor] - Monad))) - - (def: (wrap a) - (lambda [state] - (:: Monad wrap [state a]))) - - (def: (apply sFf sFa) - (lambda [state] - (do Monad - [[state f] (sFf state) - [state a] (sFa state)] - (wrap [state (f a)]))))) - -(type: #export (State' M s a) - {#;doc "Stateful computations decorated by a monad."} - (-> s (M [s a]))) - -(def: #export (run' state action) - {#;doc "Run a stateful computation decorated by a monad."} - (All [M s a] (-> s (State' M s a) (M [s a]))) - (action state)) - -(struct: #export (StateT Monad) - {#;doc "A monad transformer to create composite stateful computations."} - (All [M s] (-> (Monad M) (Monad (State' M s)))) - (def: applicative (Applicative Monad)) - (def: (join sMsMa) - (lambda [state] - (do Monad - [[state' sMa] (sMsMa state)] - (sMa state'))))) - -(def: #export (lift-state Monad ma) - {#;doc "Lift monadic values to the State' wrapper."} - (All [M s a] (-> (Monad M) (M a) (State' M s a))) - (lambda [state] - (do Monad - [a ma] - (wrap [state a])))) diff --git a/stdlib/source/lux/codata/thunk.lux b/stdlib/source/lux/codata/thunk.lux deleted file mode 100644 index 12af1dfe2..000000000 --- a/stdlib/source/lux/codata/thunk.lux +++ /dev/null @@ -1,33 +0,0 @@ -(;module: - lux - (lux [io] - (control monad) - (concurrency ["A" atom]) - [compiler] - (macro ["s" syntax #+ syntax:]))) - -(type: #export (Thunk a) - (-> [] a)) - -(def: #hidden (freeze' generator) - (All [a] (-> (-> [] a) (-> [] a))) - (let [cache (: (A;Atom (Maybe ($ +0))) - (A;atom #;None))] - (lambda [_] - (case (io;run (A;get cache)) - (#;Some value) - value - - _ - (let [value (generator [])] - (exec (io;run (A;compare-and-swap _ (#;Some value) cache)) - value)))))) - -(syntax: #export (freeze expr) - (do @ - [g!arg (compiler;gensym "")] - (wrap (list (` (freeze' (lambda [(~ g!arg)] (~ expr)))))))) - -(def: #export (thaw thunk) - (All [a] (-> (Thunk a) a)) - (thunk [])) diff --git a/stdlib/source/lux/concurrency/actor.lux b/stdlib/source/lux/concurrency/actor.lux index 97ce7dec3..56b40f41b 100644 --- a/stdlib/source/lux/concurrency/actor.lux +++ b/stdlib/source/lux/concurrency/actor.lux @@ -2,7 +2,7 @@ lux (lux (control monad) [io #- run] - (codata function) + function (data [error #- fail] text/format (coll [list "List/" Monoid Monad]) diff --git a/stdlib/source/lux/concurrency/frp.lux b/stdlib/source/lux/concurrency/frp.lux index f84103e3f..6d18a73bb 100644 --- a/stdlib/source/lux/concurrency/frp.lux +++ b/stdlib/source/lux/concurrency/frp.lux @@ -5,7 +5,6 @@ monad eq) [io #- run] - (codata function) (data (coll [list "L/" Monoid]) text/format) [compiler] diff --git a/stdlib/source/lux/concurrency/promise.lux b/stdlib/source/lux/concurrency/promise.lux index ef7efd923..e94aa68e5 100644 --- a/stdlib/source/lux/concurrency/promise.lux +++ b/stdlib/source/lux/concurrency/promise.lux @@ -5,7 +5,7 @@ text/format error) [io #- run] - (codata function) + function (control functor applicative monad) diff --git a/stdlib/source/lux/control/order.lux b/stdlib/source/lux/control/order.lux index 153100cff..eb2a6f81b 100644 --- a/stdlib/source/lux/control/order.lux +++ b/stdlib/source/lux/control/order.lux @@ -1,7 +1,7 @@ (;module: lux - (.. eq) - lux/codata/function) + (lux function) + (.. eq)) ## [Signatures] (sig: #export (Order a) diff --git a/stdlib/source/lux/data/bool.lux b/stdlib/source/lux/data/bool.lux index 0b8d207fc..2d89de635 100644 --- a/stdlib/source/lux/data/bool.lux +++ b/stdlib/source/lux/data/bool.lux @@ -2,8 +2,7 @@ lux (lux (control monoid eq - codec) - (codata function))) + codec))) ## [Structures] (struct: #export _ (Eq Bool) diff --git a/stdlib/source/lux/data/coll/list.lux b/stdlib/source/lux/data/coll/list.lux index 5f2ef3984..4f93bb541 100644 --- a/stdlib/source/lux/data/coll/list.lux +++ b/stdlib/source/lux/data/coll/list.lux @@ -8,8 +8,7 @@ [fold]) (data [number "Int/" Number Codec] bool - [product]) - codata/function)) + [product]))) ## [Types] ## (type: (List a) diff --git a/stdlib/source/lux/data/coll/set.lux b/stdlib/source/lux/data/coll/set.lux index 3ac6725c1..e10f20488 100644 --- a/stdlib/source/lux/data/coll/set.lux +++ b/stdlib/source/lux/data/coll/set.lux @@ -6,8 +6,7 @@ eq [hash #*]) (data (coll [dict] - [list "List/" Fold Functor])) - (codata function))) + [list "List/" Fold Functor])))) ## [Types] (type: #export (Set a) diff --git a/stdlib/source/lux/data/coll/stream.lux b/stdlib/source/lux/data/coll/stream.lux new file mode 100644 index 000000000..5cb0829e9 --- /dev/null +++ b/stdlib/source/lux/data/coll/stream.lux @@ -0,0 +1,142 @@ +(;module: + lux + (lux (control functor + monad + comonad) + [compiler #+ with-gensyms] + (macro ["s" syntax #+ syntax: Syntax]) + (data (coll [list "List/" Monad]) + bool) + (function [cont #+ @lazy Cont]))) + +## [Types] +(type: #export (Stream a) + {#;doc "An infinite stream of lazily-evaluated values."} + (Cont [a (Stream a)])) + +## [Utils] +(def: (cycle' x xs init full) + (All [a] + (-> a (List a) a (List a) (Stream a))) + (case xs + #;Nil (@lazy [x (cycle' init full init full)]) + (#;Cons x' xs') (@lazy [x (cycle' x' xs' init full)]))) + +## [Functions] +(def: #export (iterate f x) + {#;doc "Create a stream by applying a function to a value, and to its result, on and on..."} + (All [a] + (-> (-> a a) a (Stream a))) + (@lazy [x (iterate f (f x))])) + +(def: #export (repeat x) + {#;doc "Repeat a value forever."} + (All [a] + (-> a (Stream a))) + (@lazy [x (repeat x)])) + +(def: #export (cycle xs) + {#;doc "Go over the elements of a list forever. + + The list shouldn't be empty."} + (All [a] + (-> (List a) (Maybe (Stream a)))) + (case xs + #;Nil #;None + (#;Cons x xs') (#;Some (cycle' x xs' x xs')))) + +(do-template [ ] + [(def: #export ( s) + (All [a] (-> (Stream a) )) + (let [[h t] (cont;run s)] + ))] + + [head a h] + [tail (Stream a) t]) + +(def: #export (nth idx s) + (All [a] (-> Nat (Stream a) a)) + (let [[h t] (cont;run s)] + (if (n.> +0 idx) + (nth (n.dec idx) t) + h))) + +(do-template [ ] + [(def: #export ( pred xs) + (All [a] + (-> (Stream a) (List a))) + (let [[x xs'] (cont;run xs)] + (if + (list& x ( xs')) + (list)))) + + (def: #export ( pred xs) + (All [a] + (-> (Stream a) (Stream a))) + (let [[x xs'] (cont;run xs)] + (if + ( xs') + xs))) + + (def: #export ( pred xs) + (All [a] + (-> (Stream a) [(List a) (Stream a)])) + (let [[x xs'] (cont;run xs)] + (if + (let [[tail next] ( xs')] + [(#;Cons [x tail]) next]) + [(list) xs])))] + + [take-while drop-while split-while (-> a Bool) (pred x) pred] + [take drop split Nat (n.> +0 pred) (n.dec pred)] + ) + +(def: #export (unfold step init) + {#;doc "A stateful way of infinitely calculating the values of a stream."} + (All [a b] + (-> (-> a [a b]) a (Stream b))) + (let [[next x] (step init)] + (@lazy [x (unfold step next)]))) + +(def: #export (filter p xs) + (All [a] (-> (-> a Bool) (Stream a) (Stream a))) + (let [[x xs'] (cont;run xs)] + (if (p x) + (@lazy [x (filter p xs')]) + (filter p xs')))) + +(def: #export (partition p xs) + {#;doc "Split a stream in two based on a predicate. + + The left side contains all entries for which the predicate is true. + + The right side contains all entries for which the predicate is false."} + (All [a] (-> (-> a Bool) (Stream a) [(Stream a) (Stream a)])) + [(filter p xs) (filter (complement p) xs)]) + +## [Structures] +(struct: #export _ (Functor Stream) + (def: (map f fa) + (let [[h t] (cont;run fa)] + (@lazy [(f h) (map f t)])))) + +(struct: #export _ (CoMonad Stream) + (def: functor Functor) + (def: unwrap head) + (def: (split wa) + (let [[head tail] (cont;run wa)] + (@lazy [wa (split tail)])))) + +## [Pattern-matching] +(syntax: #export (^stream& [patterns (s;form (s;many s;any))] body [branches (s;some s;any)]) + {#;doc (doc "Allows destructuring of streams in pattern-matching expressions." + "Caveat emptor: Only use it for destructuring, and not for testing values within the streams." + (let [(^stream& x y z _tail) (some-stream-func 1 2 3)] + (func x y z)))} + (with-gensyms [g!s] + (let [body+ (` (let [(~@ (List/join (List/map (lambda [pattern] + (list (` [(~ pattern) (~ g!s)]) + (` (cont;run (~ g!s))))) + patterns)))] + (~ body)))] + (wrap (list& g!s body+ branches))))) diff --git a/stdlib/source/lux/data/format/json.lux b/stdlib/source/lux/data/format/json.lux index 8565fb32f..066777fdf 100644 --- a/stdlib/source/lux/data/format/json.lux +++ b/stdlib/source/lux/data/format/json.lux @@ -19,7 +19,6 @@ (coll [list "" Fold "List/" Monad] [vector #+ Vector vector "Vector/" Monad] [dict #+ Dict])) - (codata [function]) [compiler #+ Monad with-gensyms] (macro [syntax #+ syntax:] [ast] diff --git a/stdlib/source/lux/function.lux b/stdlib/source/lux/function.lux new file mode 100644 index 000000000..cddf5d472 --- /dev/null +++ b/stdlib/source/lux/function.lux @@ -0,0 +1,20 @@ +(;module: + lux + (lux (control monoid))) + +## [Functions] +(def: #export (const c) + {#;doc "Create constant functions."} + (All [a b] (-> a (-> b a))) + (lambda [_] c)) + +(def: #export (flip f) + {#;doc "Flips the order of the arguments of a function."} + (All [a b c] + (-> (-> a b c) (-> b a c))) + (lambda [x y] (f y x))) + +## [Structures] +(struct: #export Monoid (Monoid (All [a] (-> a a))) + (def: unit id) + (def: append .)) diff --git a/stdlib/source/lux/function/cont.lux b/stdlib/source/lux/function/cont.lux new file mode 100644 index 000000000..f6330cbe4 --- /dev/null +++ b/stdlib/source/lux/function/cont.lux @@ -0,0 +1,69 @@ +(;module: + lux + (lux (macro [ast]) + (control functor + applicative + monad) + (data (coll list)) + function)) + +## [Types] +(type: #export (Cont a) + {#;doc "Delimited continuations."} + (All [b] + (-> (-> a b) b))) + +## [Syntax] +(macro: #export (@lazy tokens state) + {#;doc (doc "Delays the evaluation of an expression, by wrapping it in a continuation 'thunk'." + (@lazy (some-computation some-input)))} + (case tokens + (^ (list value)) + (let [blank (ast;symbol ["" ""])] + (#;Right [state (list (` (;lambda [(~ blank)] ((~ blank) (~ value)))))])) + + _ + (#;Left "Wrong syntax for @lazy"))) + +## [Functions] +(def: #export (call/cc f) + {#;doc "Call with current continuation."} + (All [a b c] + (-> (-> (-> a (Cont b c)) + (Cont a c)) + (Cont a c))) + (lambda [k] + (f (lambda [a _] + (k a)) + k))) + +(def: #export (continue f thunk) + {#;doc "Forces a continuation thunk to be evaluated."} + (All [i o] + (-> (-> i o) (Cont i o) o)) + (thunk f)) + +(def: #export (run thunk) + {#;doc "Forces a continuation thunk to be evaluated."} + (All [a] + (-> (Cont a) a)) + (continue id thunk)) + +## [Structs] +(struct: #export _ (Functor Cont) + (def: (map f ma) + (lambda [k] (ma (. k f))))) + +(struct: #export _ (Applicative Cont) + (def: functor Functor) + + (def: (wrap a) + (@lazy a)) + + (def: (apply ff fa) + (@lazy ((run ff) (run fa))))) + +(struct: #export _ (Monad Cont) + (def: applicative Applicative) + + (def: join run)) diff --git a/stdlib/source/lux/function/reader.lux b/stdlib/source/lux/function/reader.lux new file mode 100644 index 000000000..955b4bba3 --- /dev/null +++ b/stdlib/source/lux/function/reader.lux @@ -0,0 +1,63 @@ +(;module: + lux + (lux (control functor + applicative + ["M" monad #*]))) + +## [Types] +(type: #export (Reader r a) + {#;doc "Computations that have access to some environmental value."} + (-> r a)) + +## [Structures] +(struct: #export Functor (All [r] (Functor (Reader r))) + (def: (map f fa) + (lambda [env] + (f (fa env))))) + +(struct: #export Applicative (All [r] (Applicative (Reader r))) + (def: functor Functor) + + (def: (wrap x) + (lambda [env] x)) + + (def: (apply ff fa) + (lambda [env] + ((ff env) (fa env))))) + +(struct: #export Monad (All [r] (Monad (Reader r))) + (def: applicative Applicative) + + (def: (join mma) + (lambda [env] + (mma env env)))) + +## [Values] +(def: #export ask + {#;doc "Get the environment."} + (All [r] (Reader r r)) + (lambda [env] env)) + +(def: #export (local change reader-proc) + {#;doc "Run computation with a locally-modified environment."} + (All [r a] (-> (-> r r) (Reader r a) (Reader r a))) + (|>. change reader-proc)) + +(def: #export (run env reader-proc) + (All [r a] (-> r (Reader r a) a)) + (reader-proc env)) + +(struct: #export (ReaderT Monad) + {#;doc "Monad transformer for Reader."} + (All [M] (-> (Monad M) (All [e] (Monad (All [a] (Reader e (M a))))))) + (def: applicative (compA Applicative (get@ #M;applicative Monad))) + (def: (join eMeMa) + (lambda [env] + (do Monad + [eMa (run env eMeMa)] + (run env eMa))))) + +(def: #export lift-reader + {#;doc "Lift monadic values to the Reader wrapper."} + (All [M e a] (-> (M a) (Reader e (M a)))) + (:: Monad wrap)) diff --git a/stdlib/source/lux/function/state.lux b/stdlib/source/lux/function/state.lux new file mode 100644 index 000000000..5f5b96e4b --- /dev/null +++ b/stdlib/source/lux/function/state.lux @@ -0,0 +1,125 @@ +(;module: + lux + (lux (control functor + ["A" applicative #*] + ["M" monad #*]))) + +## [Types] +(type: #export (State s a) + {#;doc "Stateful computations."} + (-> s [s a])) + +## [Structures] +(struct: #export Functor (All [s] (Functor (State s))) + (def: (map f ma) + (lambda [state] + (let [[state' a] (ma state)] + [state' (f a)])))) + +(struct: #export Applicative (All [s] (Applicative (State s))) + (def: functor Functor) + + (def: (wrap a) + (lambda [state] + [state a])) + + (def: (apply ff fa) + (lambda [state] + (let [[state' f] (ff state) + [state'' a] (fa state')] + [state'' (f a)])))) + +(struct: #export Monad (All [s] (Monad (State s))) + (def: applicative Applicative) + + (def: (join mma) + (lambda [state] + (let [[state' ma] (mma state)] + (ma state'))))) + +## [Values] +(def: #export get + {#;doc "Read the current state."} + (All [s] (State s s)) + (lambda [state] + [state state])) + +(def: #export (put new-state) + {#;doc "Set the new state."} + (All [s] (-> s (State s Unit))) + (lambda [state] + [new-state []])) + +(def: #export (update change) + {#;doc "Compute the new state."} + (All [s] (-> (-> s s) (State s Unit))) + (lambda [state] + [(change state) []])) + +(def: #export (use user) + {#;doc "Run function on current state."} + (All [s a] (-> (-> s a) (State s a))) + (lambda [state] + [state (user state)])) + +(def: #export (local change action) + {#;doc "Run computation with a locally-modified state."} + (All [s a] (-> (-> s s) (State s a) (State s a))) + (lambda [state] + (let [[state' output] (action (change state))] + [state output]))) + +(def: #export (run state action) + {#;doc "Run a stateful computation."} + (All [s a] (-> s (State s a) [s a])) + (action state)) + +(struct: (Functor Functor) + (All [M s] (-> (Functor M) (Functor (All [a] (-> s (M [s a])))))) + (def: (map f sfa) + (lambda [state] + (:: Functor map (lambda [[s a]] [s (f a)]) + (sfa state))))) + +(struct: (Applicative Monad) + (All [M s] (-> (Monad M) (Applicative (All [a] (-> s (M [s a])))))) + (def: functor (Functor (get@ [#M;applicative #A;functor] + Monad))) + + (def: (wrap a) + (lambda [state] + (:: Monad wrap [state a]))) + + (def: (apply sFf sFa) + (lambda [state] + (do Monad + [[state f] (sFf state) + [state a] (sFa state)] + (wrap [state (f a)]))))) + +(type: #export (State' M s a) + {#;doc "Stateful computations decorated by a monad."} + (-> s (M [s a]))) + +(def: #export (run' state action) + {#;doc "Run a stateful computation decorated by a monad."} + (All [M s a] (-> s (State' M s a) (M [s a]))) + (action state)) + +(struct: #export (StateT Monad) + {#;doc "A monad transformer to create composite stateful computations."} + (All [M s] (-> (Monad M) (Monad (State' M s)))) + (def: applicative (Applicative Monad)) + (def: (join sMsMa) + (lambda [state] + (do Monad + [[state' sMa] (sMsMa state)] + (sMa state'))))) + +(def: #export (lift-state Monad ma) + {#;doc "Lift monadic values to the State' wrapper."} + (All [M s a] (-> (Monad M) (M a) (State' M s a))) + (lambda [state] + (do Monad + [a ma] + (wrap [state a])))) diff --git a/stdlib/source/lux/function/thunk.lux b/stdlib/source/lux/function/thunk.lux new file mode 100644 index 000000000..12af1dfe2 --- /dev/null +++ b/stdlib/source/lux/function/thunk.lux @@ -0,0 +1,33 @@ +(;module: + lux + (lux [io] + (control monad) + (concurrency ["A" atom]) + [compiler] + (macro ["s" syntax #+ syntax:]))) + +(type: #export (Thunk a) + (-> [] a)) + +(def: #hidden (freeze' generator) + (All [a] (-> (-> [] a) (-> [] a))) + (let [cache (: (A;Atom (Maybe ($ +0))) + (A;atom #;None))] + (lambda [_] + (case (io;run (A;get cache)) + (#;Some value) + value + + _ + (let [value (generator [])] + (exec (io;run (A;compare-and-swap _ (#;Some value) cache)) + value)))))) + +(syntax: #export (freeze expr) + (do @ + [g!arg (compiler;gensym "")] + (wrap (list (` (freeze' (lambda [(~ g!arg)] (~ expr)))))))) + +(def: #export (thaw thunk) + (All [a] (-> (Thunk a) a)) + (thunk [])) diff --git a/stdlib/source/lux/host.jvm.lux b/stdlib/source/lux/host.jvm.lux index 002208fd2..a527a7dda 100644 --- a/stdlib/source/lux/host.jvm.lux +++ b/stdlib/source/lux/host.jvm.lux @@ -3,7 +3,6 @@ (lux (control monad [enum]) [io #+ IO Monad io] - (codata function) (data (coll [list #* "" Functor Fold "List/" Monad Monoid] [array #+ Array]) number -- cgit v1.2.3