aboutsummaryrefslogtreecommitdiff
path: root/stdlib/source
diff options
context:
space:
mode:
Diffstat (limited to 'stdlib/source')
-rw-r--r--stdlib/source/lux/control/monad/indexed.lux61
1 files changed, 61 insertions, 0 deletions
diff --git a/stdlib/source/lux/control/monad/indexed.lux b/stdlib/source/lux/control/monad/indexed.lux
new file mode 100644
index 000000000..dcc3b2067
--- /dev/null
+++ b/stdlib/source/lux/control/monad/indexed.lux
@@ -0,0 +1,61 @@
+(.module:
+ lux
+ (lux (control [monad]
+ ["p" parser])
+ (data (coll [list "list/" Functor<List> Fold<List>]))
+ [macro]
+ (macro ["s" syntax #+ Syntax syntax:])))
+
+(sig: #export (IxMonad m)
+ (: (All [p a]
+ (-> a (m p p a)))
+ wrap)
+
+ (: (All [ii it io vi vo]
+ (-> (-> vi (m it io vo))
+ (m ii it vi)
+ (m ii io vo)))
+ bind))
+
+(type: Binding [Code Code])
+
+(def: binding
+ (Syntax Binding)
+ (p.seq s.any s.any))
+
+(type: Context
+ (#Let (List Binding))
+ (#Bind Binding))
+
+(def: context
+ (Syntax Context)
+ (p.alt (p.after (s.this (' #let))
+ (s.tuple (p.some binding)))
+ binding))
+
+(def: (pair-list [binding value])
+ (All [a] (-> [a a] (List a)))
+ (list binding value))
+
+(syntax: #export (do monad
+ [context (s.tuple (p.some context))]
+ expression)
+ (macro.with-gensyms [g!bind]
+ (wrap (list (` (let [(~' @) (~ monad)
+ {#..wrap (~' wrap)
+ #..bind (~ g!bind)} (~' @)]
+ (~ (list/fold (function [context next]
+ (case context
+ (#Let bindings)
+ (` (let [(~+ (|> bindings
+ (list/map pair-list)
+ list.concat))]
+ (~ next)))
+
+ (#Bind [binding value])
+ (` ((~ g!bind)
+ (.function [(~ binding)]
+ (~ next))
+ (~ value)))))
+ expression
+ (list.reverse context)))))))))