aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--stdlib/source/lux.lux260
1 files changed, 178 insertions, 82 deletions
diff --git a/stdlib/source/lux.lux b/stdlib/source/lux.lux
index c5d363874..ac1289395 100644
--- a/stdlib/source/lux.lux
+++ b/stdlib/source/lux.lux
@@ -7,52 +7,88 @@
(_lux_def Bool
(+12 ["lux" "Bool"]
(+0 "java.lang.Boolean" (+0)))
- (+1 [["lux" "type?"] (+0 true)] (+1 [["lux" "export?"] (+0 true)] (+0))))
+ (+1 [["lux" "type?"] (+0 true)]
+ (+1 [["lux" "export?"] (+0 true)]
+ (+1 [["lux" "doc"] (+6 "Your standard, run-of-the-mill boolean values.")]
+ (+0)))))
(_lux_def Nat
(+12 ["lux" "Nat"]
(+0 "#Nat" (+0)))
- (+1 [["lux" "type?"] (+0 true)] (+1 [["lux" "export?"] (+0 true)] (+0))))
+ (+1 [["lux" "type?"] (+0 true)]
+ (+1 [["lux" "export?"] (+0 true)]
+ (+1 [["lux" "doc"] (+6 "Unsigned integers, or natural numbers.
+
+ They start at zero (+0) and extend in the positive direction.")]
+ (+0)))))
(_lux_def Int
(+12 ["lux" "Int"]
(+0 "java.lang.Long" (+0)))
- (+1 [["lux" "type?"] (+0 true)] (+1 [["lux" "export?"] (+0 true)] (+0))))
+ (+1 [["lux" "type?"] (+0 true)]
+ (+1 [["lux" "export?"] (+0 true)]
+ (+1 [["lux" "doc"] (+6 "Your standard, run-of-the-mill integer numbers.")]
+ (+0)))))
(_lux_def Real
(+12 ["lux" "Real"]
(+0 "java.lang.Double" (+0)))
- (+1 [["lux" "type?"] (+0 true)] (+1 [["lux" "export?"] (+0 true)] (+0))))
+ (+1 [["lux" "type?"] (+0 true)]
+ (+1 [["lux" "export?"] (+0 true)]
+ (+1 [["lux" "doc"] (+6 "Your standard, run-of-the-mill floating-point numbers.")]
+ (+0)))))
(_lux_def Frac
(+12 ["lux" "Frac"]
(+0 "#Frac" (+0)))
- (+1 [["lux" "type?"] (+0 true)] (+1 [["lux" "export?"] (+0 true)] (+0))))
+ (+1 [["lux" "type?"] (+0 true)]
+ (+1 [["lux" "export?"] (+0 true)]
+ (+1 [["lux" "doc"] (+6 "Fractional numbers that live in the interval [0,1).
+
+ Useful for probability, and other domains that work within that interval.")]
+ (+0)))))
(_lux_def Char
(+12 ["lux" "Char"]
(+0 "java.lang.Character" (+0)))
- (+1 [["lux" "type?"] (+0 true)] (+1 [["lux" "export?"] (+0 true)] (+0))))
+ (+1 [["lux" "type?"] (+0 true)]
+ (+1 [["lux" "export?"] (+0 true)]
+ (+1 [["lux" "doc"] (+6 "Your standard, run-of-the-mill character values.")]
+ (+0)))))
(_lux_def Text
(+12 ["lux" "Text"]
(+0 "java.lang.String" (+0)))
- (+1 [["lux" "type?"] (+0 true)] (+1 [["lux" "export?"] (+0 true)] (+0))))
+ (+1 [["lux" "type?"] (+0 true)]
+ (+1 [["lux" "export?"] (+0 true)]
+ (+1 [["lux" "doc"] (+6 "Your standard, run-of-the-mill string values.")]
+ (+0)))))
(_lux_def Void
(+12 ["lux" "Void"]
(+1))
- (+1 [["lux" "type?"] (+0 true)] (+1 [["lux" "export?"] (+0 true)] (+0))))
+ (+1 [["lux" "type?"] (+0 true)]
+ (+1 [["lux" "export?"] (+0 true)]
+ (+1 [["lux" "doc"] (+6 "An unusual type that possesses no value, and thus can't be instantiated.")]
+ (+0)))))
(_lux_def Unit
(+12 ["lux" "Unit"]
(+2))
- (+1 [["lux" "type?"] (+0 true)] (+1 [["lux" "export?"] (+0 true)] (+0))))
+ (+1 [["lux" "type?"] (+0 true)]
+ (+1 [["lux" "export?"] (+0 true)]
+ (+1 [["lux" "doc"] (+6 "An unusual type that only possesses a single value: []")]
+ (+0)))))
(_lux_def Ident
(+12 ["lux" "Ident"]
(+4 Text Text))
- (+1 [["lux" "type?"] (+0 true)] (+1 [["lux" "export?"] (+0 true)] (+0))))
+ (+1 [["lux" "type?"] (+0 true)]
+ (+1 [["lux" "export?"] (+0 true)]
+ (+1 [["lux" "doc"] (+6 "An identifier.
+
+ It is used as part of Lux syntax to represent symbols and tags.")]
+ (+0)))))
## (type: (List a)
## #Nil
@@ -69,7 +105,8 @@
(+1 [["lux" "export?"] (+0 true)]
(+1 [["lux" "tags"] (+8 (+1 (+6 "Nil") (+1 (+6 "Cons") (+0))))]
(+1 [["lux" "type-args"] (+8 (+1 (+6 "a") (+0)))]
- (+0))))))
+ (+1 [["lux" "doc"] (+6 "A potentially empty list of values.")]
+ (+0)))))))
## (type: (Maybe a)
## #None
@@ -85,7 +122,8 @@
(#Cons [["lux" "export?"] (+0 true)]
(#Cons [["lux" "tags"] (+8 (#Cons (+6 "None") (#Cons (+6 "Some") #Nil)))]
(#Cons [["lux" "type-args"] (+8 (#Cons (+6 "a") #Nil))]
- #Nil)))))
+ (#Cons [["lux" "doc"] (+6 "A potentially missing value.")]
+ #Nil))))))
## (type: #rec Type
## (#HostT Text (List Type))
@@ -166,7 +204,8 @@
(#Cons [["lux" "type?"] (+0 true)]
(#Cons [["lux" "export?"] (+0 true)]
(#Cons [["lux" "doc"] (+6 "The type of things whose type doesn't matter.
- It can be used to write functions or data-structures that can take, or return anything.")]
+
+ It can be used to write functions or data-structures that can take, or return, anything.")]
#Nil))))
## (type: Bottom
@@ -177,7 +216,8 @@
(#Cons [["lux" "type?"] (+0 true)]
(#Cons [["lux" "export?"] (+0 true)]
(#Cons [["lux" "doc"] (+6 "The type of things whose type is unknown or undefined.
- Useful for expressions that cause errors or other \"extraordinary\" conditions.")]
+
+ Useful for expressions that cause errors or other \"extraordinary\" conditions.")]
#Nil))))
## (type: #rec Ann-Value
@@ -233,7 +273,8 @@
(#Cons (+6 "DictM")
#Nil)))))))))))]
(#Cons [["lux" "type-rec?"] (+0 true)]
- #Nil)))))
+ (#Cons [["lux" "doc"] (+6 "The value of an individual annotation.")]
+ #Nil))))))
## (type: Anns
## (List [Ident Ann-Value]))
@@ -242,7 +283,8 @@
(#AppT List (#ProdT Ident Ann-Value)))
(#Cons [["lux" "type?"] (#BoolM true)]
(#Cons [["lux" "export?"] (#BoolM true)]
- #Nil)))
+ (#Cons [["lux" "doc"] (#TextM "A set of annotations associated with a definition.")]
+ #Nil))))
(_lux_def default-def-meta-exported
(_lux_: Anns
@@ -258,11 +300,12 @@
#Nil)
## (type: Def
-## [Type Anns Unit])
+## [Type Anns Void])
(_lux_def Def
(#NamedT ["lux" "Def"]
- (#ProdT Type (#ProdT Anns Unit)))
- default-def-meta-exported)
+ (#ProdT Type (#ProdT Anns Void)))
+ (#Cons [["lux" "doc"] (#TextM "Represents all the data associated with a definition: its type, its annotations, and its value.")]
+ default-def-meta-exported))
## (type: (Bindings k v)
## {#counter Nat
@@ -435,7 +478,8 @@
(#Cons (#TextM "Right")
#Nil)))]
(#Cons [["lux" "type-args"] (#ListM (#Cons (#TextM "l") (#Cons (#TextM "r") #;Nil)))]
- default-def-meta-exported)))
+ (#Cons [["lux" "doc"] (#TextM "A choice between two values of different types.")]
+ default-def-meta-exported))))
## (type: Source
## (List (Meta Cursor Text)))
@@ -490,15 +534,16 @@
(#Cons (#TextM "types")
(#Cons (#TextM "module-anns")
#Nil))))))))]
- default-def-meta-exported))
+ (#Cons [["lux" "doc"] (#TextM "All the information contained within a Lux module.")]
+ default-def-meta-exported)))
-## (type: CompilerMode
+## (type: Compiler-Mode
## #Release
## #Debug
## #Eval
## #REPL)
-(_lux_def CompilerMode
- (#NamedT ["lux" "CompilerMode"]
+(_lux_def Compiler-Mode
+ (#NamedT ["lux" "Compiler-Mode"]
(#SumT ## "lux;Release"
#UnitT
(#SumT ## "lux;Debug"
@@ -512,28 +557,30 @@
(#Cons (#TextM "Eval")
(#Cons (#TextM "REPL")
#Nil)))))]
- default-def-meta-exported))
+ (#Cons [["lux" "doc"] (#TextM "A sign that shows the conditions under which the compiler is running.")]
+ default-def-meta-exported)))
-## (type: CompilerInfo
+## (type: Compiler-Info
## {#compiler-name Text
## #compiler-version Text
-## #compiler-mode CompilerMode})
-(_lux_def CompilerInfo
- (#NamedT ["lux" "CompilerInfo"]
+## #compiler-mode Compiler-Mode})
+(_lux_def Compiler-Info
+ (#NamedT ["lux" "Compiler-Info"]
(#ProdT ## "lux;compiler-name"
Text
(#ProdT ## "lux;compiler-version"
Text
## "lux;compiler-mode"
- CompilerMode)))
+ Compiler-Mode)))
(#Cons [["lux" "tags"] (#ListM (#Cons (#TextM "compiler-name")
(#Cons (#TextM "compiler-version")
(#Cons (#TextM "compiler-mode")
#Nil))))]
- default-def-meta-exported))
+ (#Cons [["lux" "doc"] (#TextM "Information about the current version and type of compiler that is running.")]
+ default-def-meta-exported)))
## (type: Compiler
-## {#info CompilerInfo
+## {#info Compiler-Info
## #source Source
## #cursor Cursor
## #modules (List [Text Module])
@@ -546,7 +593,7 @@
(_lux_def Compiler
(#NamedT ["lux" "Compiler"]
(#ProdT ## "lux;info"
- CompilerInfo
+ Compiler-Info
(#ProdT ## "lux;source"
Source
(#ProdT ## "lux;cursor"
@@ -578,7 +625,8 @@
(#Cons (#TextM "host")
#Nil)))))))))))]
(#Cons [["lux" "doc"] (#TextM "Represents the state of the Lux compiler during a run.
- It's provided to macros during their invocation, so they can access compiler data.
+
+ It is provided to macros during their invocation, so they can access compiler data.
Caveat emptor: Avoid fiddling with it, unless you know what you're doing.")]
default-def-meta-exported)))
@@ -601,7 +649,8 @@
(_lux_def Macro
(#NamedT ["lux" "Macro"]
(#LambdaT ASTList (#AppT Lux ASTList)))
- default-def-meta-exported)
+ (#Cons [["lux" "doc"] (#TextM "Functions that run at compile-time and allow you to transform and extend the language in powerful ways.")]
+ default-def-meta-exported))
## Base functions & macros
## (def: _cursor
@@ -923,7 +972,7 @@
(macro:' #export (comment tokens)
(#Cons [["lux" "doc"] (#TextM "## Throws away any code given to it.
- ## Great for commenting out code, while retaining syntax high-lightning and formatting in your text editor.
+ ## Great for commenting-out code, while retaining syntax high-lighting and formatting in your text editor.
(comment 1 2 3 4)")]
#;Nil)
(return #Nil))
@@ -1571,9 +1620,13 @@
)))
(macro:' #export (if tokens)
- (list [["lux" "doc"] (#TextM "(if true
+ (list [["lux" "doc"] (#TextM "Picks which expression to evaluate based on a boolean test value.
+
+ (if true
\"Oh, yeah!\"
- \"Aw hell naw!\")")])
+ \"Aw hell naw!\")
+
+ == \"Oh, yeah!\"")])
(_lux_case tokens
(#Cons test (#Cons then (#Cons else #Nil)))
(return (list (form$ (list (symbol$ ["" "_lux_case"]) test
@@ -2142,7 +2195,11 @@
(i.= 0 (i.% div n)))
(def:''' #export (not x)
- #Nil
+ (list [["lux" "doc"] (#TextM "## Boolean negation.
+
+ (not true) == false
+
+ (not false) == true")])
(-> Bool Bool)
(if x false true))
@@ -2664,7 +2721,7 @@
(macro:' #export (^ tokens)
(list [["lux" "doc"] (#TextM "## Macro-expanding patterns.
- ## It's a special macro meant to be used with case.
+ ## It's a special macro meant to be used with 'case'.
(case (: (List Int) (list 1 2 3))
(^ (list x y z))
(#Some ($_ i.* x y z))
@@ -2687,15 +2744,15 @@
(macro:' #export (^or tokens)
(list [["lux" "doc"] (#TextM "## Or-patterns.
- ## It's a special macro meant to be used with case.
+ ## It's a special macro meant to be used with 'case'.
(type: Weekday
- (| #Monday
- #Tuesday
- #Wednesday
- #Thursday
- #Friday
- #Saturday
- #Sunday))
+ #Monday
+ #Tuesday
+ #Wednesday
+ #Thursday
+ #Friday
+ #Saturday
+ #Sunday)
(def: (weekend? day)
(-> Weekday Bool)
@@ -2913,7 +2970,10 @@
(#;Some (#;Right []))
(list (' #hidden))))
-(def:' #export (log! message)
+(def:''' #export (log! message)
+ (list [["lux" "doc"] (#TextM "Logs message to standard output.
+
+ Useful for debugging.")])
(-> Text Unit)
(_lux_proc ["jvm" "invokevirtual:java.io.PrintStream:println:java.lang.String"]
[(_lux_proc ["jvm" "getstatic:java.lang.System:out"] []) message]))
@@ -3009,7 +3069,9 @@
base))
(macro:' #export (macro: tokens)
- (list [["lux" "doc"] (#TextM "(macro: #export (ident-for tokens)
+ (list [["lux" "doc"] (#TextM "Macro-definition macro.
+
+ (macro: #export (ident-for tokens)
(case tokens
(^template [<tag>]
(^ (list [_ (<tag> [prefix name])]))
@@ -3485,7 +3547,9 @@
(fail "Wrong syntax for struct:"))))
(def: #export (id x)
- {#;doc "Identity function. Does nothing to it's argument and just returns it."}
+ {#;doc "Identity function.
+
+ Does nothing to it's argument and just returns it."}
(All [a] (-> a a))
x)
@@ -3502,8 +3566,8 @@
_
(fail <message>)))]
- [and (if (~ pre) (~ post) false) "'and' requires >=1 clauses." "Short-circuiting \"and\"\n(and true false true) ## => false"]
- [or (if (~ pre) true (~ post)) "'or' requires >=1 clauses." "Short-circuiting \"or\"\n(or true false true) ## => true"])
+ [and (if (~ pre) (~ post) false) "'and' requires >=1 clauses." "Short-circuiting \"and\".\n(and true false true) ## => false"]
+ [or (if (~ pre) true (~ post)) "'or' requires >=1 clauses." "Short-circuiting \"or\".\n(or true false true) ## => true"])
(macro: #export (type: tokens)
{#;doc "## The type-definition macro.
@@ -3971,7 +4035,7 @@
(#Some def-type)))))
(def: (find-def-value name state)
- (-> Ident (Lux [Type Unit]))
+ (-> Ident (Lux [Type Void]))
(let [[v-prefix v-name] name
{#info info #source source #modules modules
#scopes scopes #type-vars types #host host
@@ -4136,8 +4200,8 @@
(macro: #export (cond tokens)
{#;doc "## Branching structures with multiple test conditions.
- (cond (even? num) \"even\"
- (odd? num) \"odd\"
+ (cond (n.even? num) \"even\"
+ (n.odd? num) \"odd\"
## else-branch
\"???\")"}
(if (i.= 0 (i.% 2 (length tokens)))
@@ -4289,7 +4353,7 @@
(macro: #export (default tokens state)
{#;doc "## Allows you to provide a default value that will be used
- ## if a (Maybe x) value turns out to be #;Some.
+ ## if a (Maybe x) value turns out to be #;None.
(default 20 (#;Some 10)) => 10
(default 20 #;None) => 20"}
@@ -4382,7 +4446,7 @@
(wrap (List/append defs openings))
))
-(macro: #export (refer tokens)
+(macro: #hidden (refer tokens)
(case tokens
(^ (list& [_ (#TextS module-name)] options))
(do Monad<Lux>
@@ -4417,7 +4481,11 @@
(~' #open) ((~@ =opens))))))
(macro: #export (module: tokens)
- {#;doc "## Examples
+ {#;doc "Module-definition macro.
+
+ Can take optional annotations and allows the specification of modules to import.
+
+ ## Examples
(;module: {#;doc \"Some documentation...\"}
lux
(lux (control (monad #as M #refer #all))
@@ -4668,9 +4736,6 @@
(#;BoundT idx)
(default type (list;at idx env))
- (#;NamedT name type)
- (beta-reduce env type)
-
_
type
))"}
@@ -4774,15 +4839,17 @@
)]
($_ Text/append "\"" escaped "\"")))
-(do-template [<name> <op> <one> <type>]
+(do-template [<name> <op> <one> <type> <doc>]
[(def: #export (<name> value)
+ {#;doc <doc>}
(-> <type> <type>)
(<op> <one> value))]
- [i.inc i.+ 1 Int]
- [i.dec i.- 1 Int]
- [n.inc n.+ +1 Nat]
- [n.dec n.- +1 Nat])
+ [i.inc i.+ 1 Int "Increment function."]
+ [i.dec i.- 1 Int "Decrement function."]
+ [n.inc n.+ +1 Nat "Increment function."]
+ [n.dec n.- +1 Nat "Decrement function."]
+ )
(def: tag->Text
(-> Ident Text)
@@ -4879,17 +4946,16 @@
(Text/append text "\n\n"))))
(macro: #export (doc tokens)
- {#;doc "Creates code documentation, embedding text as comments and properly formatting the forms it's being given.
+ {#;doc "## Creates code documentation, embedding text as comments and properly formatting the forms it's being given.
## For Example:
- (doc
- \"Allows arbitrary looping, using the \\\"recur\\\" form to re-start the loop.
- Can be used in monadic code to create monadic loops.\"
- (loop [count 0
- x init]
- (if (< 10 count)
- (recur (i.inc count) (f x))
- x)))"}
+ (doc \"Allows arbitrary looping, using the \\\"recur\\\" form to re-start the loop.
+ Can be used in monadic code to create monadic loops.\"
+ (loop [count 0
+ x init]
+ (if (< 10 count)
+ (recur (i.inc count) (f x))
+ x)))"}
(return (list (` (#;TextM (~ (|> tokens
(map (. doc-fragment->Text identify-doc-fragment))
Text/join
@@ -5258,7 +5324,7 @@
"Useful in situations where the result of a branch depends on further refinements on the values being matched."
"For example:"
(case (split (size static) uri)
- (^=> (#;Some [chunk uri']) {(Text/= static chunk) true})
+ (^=> (#;Some [chunk uri']) [(Text/= static chunk) true])
(match-uri endpoint? parts' uri')
_
@@ -5344,6 +5410,14 @@
(list-at (n.dec idx) xs'))))
(macro: #export ($ tokens)
+ {#;doc (doc "Allows you to refer to the type-variables in a polymorphic function's type, by their index."
+ "In the example below, +0 corresponds to the 'a' variable."
+ (def: #export (from-list list)
+ (All [a] (-> (List a) (Vector a)))
+ (List/fold add
+ (: (Vector ($ +0))
+ empty)
+ list)))}
(case tokens
(^ (list [_ (#NatS idx)]))
(do Monad<Lux>
@@ -5371,10 +5445,10 @@
(macro: #export (^@ tokens)
{#;doc (doc "Allows you to simultaneously bind and de-structure a value."
- (def: (hash (^@ set [a/Hash _]))
- (List/fold (lambda [elem acc] (i.+ (:: a/Hash hash elem) acc))
- 0
- (->List set))))}
+ (def: (hash (^@ set [Hash<a> _]))
+ (List/fold (lambda [elem acc] (n.+ (:: Hash<a> hash elem) acc))
+ +0
+ (to-list set))))}
(case tokens
(^ (list& [_meta (#;FormS (list [_ (#;SymbolS ["" name])] pattern))] body branches))
(let [g!whole (symbol$ ["" name])]
@@ -5386,6 +5460,10 @@
(fail "Wrong syntax for ^@")))
(macro: #export (^|> tokens)
+ {#;doc (doc "Pipes the value being pattern-matched against prior to binding it to a variable."
+ (case input
+ (^|> value [n.inc (n.% +10) (n.max +1)])
+ (foo value)))}
(case tokens
(^ (list& [_meta (#;FormS (list [_ (#;SymbolS ["" name])] [_ (#;TupleS steps)]))] body branches))
(let [g!name (symbol$ ["" name])]
@@ -5447,9 +5525,11 @@
(macro: #export (undefined tokens)
{#;doc (doc "Meant to be used as a stand-in for functions with undefined implementations."
+ "Undefined expressions will type-check against everything, so they make good dummy implementations."
(def: (square x)
(-> Int Int)
- (undefined)))}
+ (undefined))
+ "If an undefined expression is ever evaluated, it will raise an error.")}
(case tokens
#;Nil
(return (list (` (error! (with-cursor "Undefined behavior.")))))
@@ -5458,6 +5538,11 @@
(fail "Wrong syntax for undefined")))
(macro: #export (@pre tokens)
+ {#;doc (doc "Pre-conditions."
+ "Given a test and an expression to run, only runs the expression if the test passes."
+ "Otherwise, an error is raised."
+ (@pre (i.= 4 (i.+ 2 2))
+ (foo 123 456 789)))}
(case tokens
(^ (list test expr))
(return (list (` (if (~ test)
@@ -5468,6 +5553,12 @@
(fail "Wrong syntax for @pre")))
(macro: #export (@post tokens)
+ {#;doc (doc "Post-conditions."
+ "Given a predicate and an expression to run, evaluates the expression and then tests the output with the predicate."
+ "If the predicate returns true, returns the value of the expression."
+ "Otherwise, an error is raised."
+ (@post i.even?
+ (i.+ 2 2)))}
(case tokens
(^ (list test expr))
(do Monad<Lux>
@@ -5493,6 +5584,11 @@
)
(macro: #export (type-of tokens)
+ {#;doc (doc "Generates the type corresponding to a given definition or variable."
+ (let [my-num (: Int 123)]
+ (type-of my-num))
+ "=="
+ Int)}
(case tokens
(^ (list [_ (#;SymbolS var-name)]))
(do Monad<Lux>