Reshuffled all the library code into files that make more sense and
cleaned up some of the library code.
This commit is contained in:
parent
06bf76f818
commit
e09f05a50c
@ -1,5 +1,8 @@
|
||||
#..
|
||||
This file contains code that supports manipulating and using collections like lists
|
||||
and dictionaries.
|
||||
|
||||
use "lib/metaprogramming.nom"
|
||||
use "lib/utils.nom"
|
||||
use "lib/control_flow.nom"
|
||||
use "lib/operators.nom"
|
||||
|
||||
@ -148,6 +151,13 @@ immediately:
|
||||
return comprehension;
|
||||
end)(nomsu)
|
||||
|
||||
# Sorting:
|
||||
compile [sort %items] to: "table.sort(\(%items as lua))"
|
||||
compile [sort %items by %key_expr] to: ".."
|
||||
nomsu.utils.sort(\(%items as lua), function(\(\% as lua))
|
||||
return \(%key_expr as lua);
|
||||
end)
|
||||
|
||||
action [%items sorted]:
|
||||
%copy = (% for all %items)
|
||||
sort %copy
|
||||
@ -156,6 +166,7 @@ action [%items sorted by %key]:
|
||||
%copy = (% for all %items)
|
||||
sort %copy by %key
|
||||
return %copy
|
||||
|
||||
action [unique %items]:
|
||||
[%k for %k=%v in {%=(yes) for all %items}]
|
||||
|
||||
|
@ -1,6 +1,19 @@
|
||||
#..
|
||||
This file contains compile-time actions that define basic control flow structures
|
||||
like "if" statements and loops.
|
||||
|
||||
use "lib/metaprogramming.nom"
|
||||
use "lib/text.nom"
|
||||
use "lib/operators.nom"
|
||||
use "lib/utils.nom"
|
||||
|
||||
# No-Op
|
||||
immediately:
|
||||
compile [do nothing] to code: ""
|
||||
|
||||
# Return
|
||||
immediately:
|
||||
compile [return] to code: "do return; end"
|
||||
compile [return %return_value] to code: "do return \(%return_value as lua); end"
|
||||
|
||||
# Conditionals
|
||||
immediately:
|
||||
@ -17,10 +30,33 @@ immediately:
|
||||
\(%else_body as lua statements)
|
||||
end --end if
|
||||
|
||||
# Return
|
||||
# Conditional expression (ternary operator)
|
||||
#.. Note: this uses a function instead of "(condition and if_expr or else_expr)"
|
||||
because that breaks if %if_expr is falsey, e.g. "x < 5 and false or 99"
|
||||
immediately:
|
||||
compile [return] to code: "do return; end"
|
||||
compile [return %return_value] to code: "do return \(%return_value as lua); end"
|
||||
compile [..]
|
||||
%when_true_expr if %condition else %when_false_expr
|
||||
%when_true_expr if %condition otherwise %when_false_expr
|
||||
%when_false_expr unless %condition else %when_true_expr
|
||||
%when_false_expr unless %condition then %when_true_expr
|
||||
..to:
|
||||
local %safe
|
||||
#.. If %when_true_expr is guaranteed to be truthy, we can use Lua's idiomatic
|
||||
equivalent of a conditional expression: (cond and if_true or if_false)
|
||||
if ({Text=yes, List=yes, Dict=yes, Number=yes}->(%when_true_expr's "type")):
|
||||
return "(\(%condition as lua) and \(%when_true_expr as lua) or \(%when_false_expr as lua))"
|
||||
..else:
|
||||
#.. Otherwise, need to do an anonymous inline function (yuck, too bad lua
|
||||
doesn't have a proper ternary operator!)
|
||||
To see why this is necessary consider: (random()<.5 and false or 99)
|
||||
return ".."
|
||||
(function(nomsu)
|
||||
if \(%condition as lua) then
|
||||
return \(%when_true_expr as lua);
|
||||
else
|
||||
return \(%when_false_expr as lua);
|
||||
end
|
||||
end)(nomsu)
|
||||
|
||||
# GOTOs
|
||||
immediately:
|
||||
@ -212,7 +248,7 @@ immediately:
|
||||
set %first = (yes)
|
||||
for %func_call in (%body's "value"):
|
||||
local [%tokens, %star, %condition, %action]
|
||||
assume ((%func_call's "type") == "FunctionCall") or barf ".."
|
||||
assume ((%func_call's "type") is "FunctionCall") or barf ".."
|
||||
Invalid format for 'when' statement. Only '*' blocks are allowed.
|
||||
set %tokens = (%func_call's "value")
|
||||
set %star = (%tokens -> 1)
|
||||
@ -308,7 +344,7 @@ immediately:
|
||||
..to code: ".."
|
||||
do
|
||||
local fell_through = false;
|
||||
local ok, ret1, ret2 = pcall(function(nomsu)
|
||||
local ok, ret = pcall(function(nomsu)
|
||||
\(%action as lua statements)
|
||||
fell_through = true;
|
||||
end, nomsu);
|
||||
@ -318,7 +354,7 @@ immediately:
|
||||
if not ok then
|
||||
\(%fallback as lua statements)
|
||||
elseif not fell_through then
|
||||
return ret1, ret2;
|
||||
return ret;
|
||||
end
|
||||
end
|
||||
parse [try %action] as:
|
||||
@ -332,20 +368,48 @@ immediately:
|
||||
|
||||
# Do/finally:
|
||||
immediately:
|
||||
compile [do %action] to code:
|
||||
(%action as lua statements) if ((%action's "type") is "Block")
|
||||
..else "(\(%action as lua))(nomsu);"
|
||||
|
||||
compile [do %action then always %final_action] to code: ".."
|
||||
do
|
||||
local fell_through = false;
|
||||
local ok, ret1, ret2 = pcall(function(nomsu)
|
||||
local ok, ret1 = pcall(function(nomsu)
|
||||
\(%action as lua statements)
|
||||
fell_through = true;
|
||||
end, nomsu);
|
||||
local ok2, _ = pcall(function(nomsu)
|
||||
local ok2, ret2 = pcall(function(nomsu)
|
||||
\(%final_action as lua statements)
|
||||
end, nomsu);
|
||||
if not ok then nomsu:error(ret1); end
|
||||
if not ok2 then nomsu:error(ret2); end
|
||||
if not fell_through then
|
||||
return ret1, ret2;
|
||||
return ret1;
|
||||
end
|
||||
end
|
||||
|
||||
immediately:
|
||||
compile [with %assignments %action] to code:
|
||||
local [%lua, %olds, %old_vals, %new_vals]
|
||||
set {%temp_vars=[], %old_vals=[], %new_vals=[]}
|
||||
for %i=%assignment in (%assignments' "value"):
|
||||
set (%temp_vars->%i) = "temp\%i"
|
||||
set (%old_vals->%i) = ((%assignment's "dict_key") as lua)
|
||||
set (%new_vals->%i) = ((%assignment's "dict_value") as lua)
|
||||
return ".."
|
||||
do
|
||||
local \(join %temp_vars with ", ") = \(join %old_vals with ", ");
|
||||
\(join %old_vals with ", ") = \(join %new_vals with ", ");
|
||||
local fell_through = false;
|
||||
local ok, ret = pcall(function(nomsu)
|
||||
do
|
||||
\(%action as lua statements)
|
||||
end
|
||||
fell_through = true;
|
||||
end, nomsu);
|
||||
\(join %old_vals with ", ") = \(join %temp_vars with ", ");
|
||||
if not ok then error(ret, 0); end
|
||||
if not fell_through then return ret end
|
||||
end
|
||||
|
||||
|
@ -1,7 +1,11 @@
|
||||
#..
|
||||
This file imports all the commonly used library files, which can be convenient for
|
||||
avoiding retyping the whole list.
|
||||
|
||||
use "lib/metaprogramming.nom"
|
||||
use "lib/utils.nom"
|
||||
use "lib/text.nom"
|
||||
use "lib/operators.nom"
|
||||
use "lib/control_flow.nom"
|
||||
use "lib/math.nom"
|
||||
use "lib/collections.nom"
|
||||
use "lib/utils2.nom"
|
||||
lua> "nomsu.core_defs = nomsu.__class.def_number;"
|
||||
|
79
lib/math.nom
Normal file
79
lib/math.nom
Normal file
@ -0,0 +1,79 @@
|
||||
#..
|
||||
This file defines some common math literals and functions
|
||||
|
||||
use "lib/metaprogramming.nom"
|
||||
use "lib/control_flow.nom"
|
||||
|
||||
# Literals:
|
||||
compile [infinity, inf] to: "math.huge"
|
||||
compile [not a number, NaN, nan] to: "(0/0)"
|
||||
compile [pi, Pi, PI] to: "math.pi"
|
||||
compile [tau, Tau, TAU] to: "(2*math.pi)"
|
||||
compile [golden ratio] to: "((1+math.sqrt(5))/2)"
|
||||
compile [e] to: "math.e"
|
||||
|
||||
# Functions:
|
||||
compile [% as number] to: "tonumber(\(% as lua))"
|
||||
compile [absolute value %, | % |, abs %] to: "math.abs(\(% as lua))"
|
||||
compile [square root %, √%, sqrt %] to: "math.sqrt(\(% as lua))"
|
||||
compile [sine %, sin %] to: "math.sin(\(% as lua))"
|
||||
compile [cosine %, cos %] to: "math.cos(\(% as lua))"
|
||||
compile [tangent %, tan %] to: "math.tan(\(% as lua))"
|
||||
compile [arc sine %, asin %] to: "math.asin(\(% as lua))"
|
||||
compile [arc cosine %, acos %] to: "math.acos(\(% as lua))"
|
||||
compile [arc tangent %, atan %] to: "math.atan(\(% as lua))"
|
||||
compile [arc tangent %y/%x, atan2 %y %x] to: "math.atan2(\(%y as lua), \(%x as lua))"
|
||||
compile [hyperbolic sine %, sinh %] to: "math.sinh(\(% as lua))"
|
||||
compile [hyperbolic cosine %, cosh %] to: "math.cosh(\(% as lua))"
|
||||
compile [hyperbolic tangent %, tanh %] to: "math.tanh(\(% as lua))"
|
||||
compile [e^%, exp %] to: "math.exp(\(% as lua))"
|
||||
compile [natural log %, ln %, log %] to: "math.log(\(% as lua))"
|
||||
compile [log % base %base, log_%base %, log base %base %] to: "math.log(\(% as lua), \(%base as lua))"
|
||||
compile [floor %] to: "math.floor(\(% as lua))"
|
||||
compile [ceiling %, ceil %] to: "math.ceil(\(% as lua))"
|
||||
compile [round %, % rounded] to: "math.floor(\(% as lua) + .5)"
|
||||
action [%n to the nearest %rounder]:
|
||||
=lua "(\%rounder)*math.floor((\%n / \%rounder) + .5)"
|
||||
|
||||
# Any/all/none
|
||||
compile [all of %items, all %items] to:
|
||||
"(\(join ((% as lua) for all (%items' "value")) with " and "))"
|
||||
..if ((%items' "type") is "List") else "nomsu.utils.all(\(%items as lua))"
|
||||
parse [not all of %items, not all %items] as: not (all of %items)
|
||||
compile [any of %items, any %items] to:
|
||||
"(\(join ((% as lua) for all (%items' "value")) with " or "))"
|
||||
..if ((%items' "type") is "List") else "nomsu.utils.any(\(%items as lua))"
|
||||
parse [none of %items, none %items] as: not (any of %items)
|
||||
compile [sum of %items, sum %items] to:
|
||||
"(\(join ((% as lua) for all (%items' "value")) with " + "))"
|
||||
..if ((%items' "type") is "List") else "nomsu.utils.sum(\(%items as lua))"
|
||||
compile [product of %items, product %items] to:
|
||||
"(\(join ((% as lua) for all (%items' "value")) with " * "))"
|
||||
..if ((%items' "type") is "List") else "nomsu.utils.product(\(%items as lua))"
|
||||
action [avg of %items, average of %items]:
|
||||
=lua "(nomsu.utils.sum(\%items)/#\%items)"
|
||||
compile [min of %items, smallest of %items, lowest of %items] to:
|
||||
"nomsu.utils.min(\(%items as lua))"
|
||||
compile [max of %items, biggest of %items, largest of %items, highest of %items] to:
|
||||
"nomsu.utils.max(\(%items as lua))"
|
||||
compile [min of %items by %value_expr] to: ".."
|
||||
nomsu.utils.min(\(%items as lua), function(\(\% as lua))
|
||||
return \(%value_expr as lua)
|
||||
end)
|
||||
compile [max of %items by %value_expr] to: ".."
|
||||
nomsu.utils.max(\(%items as lua), function(\(\% as lua))
|
||||
return \(%value_expr as lua)
|
||||
end)
|
||||
|
||||
# Random functions
|
||||
action [seed random with %]:
|
||||
lua> ".."
|
||||
math.randomseed(\%);
|
||||
for i=1,20 do; math.random(); end;
|
||||
parse [seed random] as: seed random with (=lua "os.time()")
|
||||
compile [random number, random, rand] to: "math.random()"
|
||||
compile [random int %n, random integer %n, randint %n] to: "math.random(\(%n as lua))"
|
||||
compile [random from %low to %high, random number from %low to %high, rand %low %high] to:
|
||||
"math.random(\(%low as lua), \(%high as lua))"
|
||||
action [random choice from %elements, random choice %elements, random %elements]:
|
||||
=lua "\%elements[math.random(#\%elements)]"
|
@ -176,3 +176,23 @@ parse [parse tree %code] as: nomsu "tree_to_str" [\%code]
|
||||
|
||||
parse [enable debugging] as: lua> "nomsu.debug = true"
|
||||
parse [disable debugging] as: lua> "nomsu.debug = false"
|
||||
|
||||
compile [say %str] to:
|
||||
lua> ".."
|
||||
if \%str.type == "Text" then
|
||||
return "nomsu:writeln("..\(%str as lua)..")";
|
||||
else
|
||||
return "nomsu:writeln(nomsu:stringify("..\(%str as lua).."))";
|
||||
end
|
||||
|
||||
# Error functions
|
||||
compile [barf!] to: "error(nil, 0)"
|
||||
compile [barf %msg] to: "error(\(%msg as lua), 0)"
|
||||
compile [assume %condition] to: "assert(\(%condition as lua))"
|
||||
compile [assume %condition or barf %msg] to: "assert(\(%condition as lua), \(%msg as lua))"
|
||||
|
||||
# Literals
|
||||
compile [yes] to: "true"
|
||||
compile [no] to: "false"
|
||||
compile [nothing, nil, null] to: "nil"
|
||||
|
||||
|
@ -1,90 +1,14 @@
|
||||
#..
|
||||
This file contains definitions of operators like "+" and "and".
|
||||
|
||||
use "lib/metaprogramming.nom"
|
||||
|
||||
# Literals
|
||||
compile [yes] to: "true"
|
||||
compile [no] to: "false"
|
||||
compile [nil, null] to: "nil"
|
||||
compile [infinity, inf] to: "math.huge"
|
||||
compile [not a number, NaN, nan] to: "(0/0)"
|
||||
compile [pi, Pi, PI] to: "math.pi"
|
||||
compile [tau, Tau, TAU] to: "(2*math.pi)"
|
||||
compile [phi, Phi, PHI, golden ratio] to: "((1+math.sqrt(5))/2)"
|
||||
compile [do nothing] to code: ""
|
||||
|
||||
# Ternary operator
|
||||
#.. Note: this uses a function instead of "(condition and if_expr or else_expr)"
|
||||
because that breaks if %if_expr is falsey, e.g. "x < 5 and false or 99"
|
||||
compile [..]
|
||||
%when_true_expr if %condition else %when_false_expr
|
||||
%when_true_expr if %condition otherwise %when_false_expr
|
||||
%when_false_expr unless %condition else %when_true_expr
|
||||
%when_false_expr unless %condition then %when_true_expr
|
||||
..to:
|
||||
lua> ".."
|
||||
local condition = nomsu:tree_to_lua(\%condition).expr;
|
||||
local when_true = nomsu:tree_to_lua(\%when_true_expr).expr;
|
||||
local when_false = nomsu:tree_to_lua(\%when_false_expr).expr;
|
||||
local safe = {Text=true, List=true, Dict=true, Number=true};
|
||||
if safe[\%when_true_expr.type] then
|
||||
return "("..condition.." and "..when_true.." or "..when_false..")";
|
||||
else
|
||||
return ([[
|
||||
(function(nomsu)
|
||||
if %s then
|
||||
return %s;
|
||||
else
|
||||
return %s;
|
||||
end
|
||||
end)(nomsu)]]):format(condition, when_true, when_false);
|
||||
end
|
||||
|
||||
# Indexing:
|
||||
immediately:
|
||||
compile [%obj'%key, %obj's %key, %obj -> %key] to: "(\(%obj as lua))[\(%key as lua)]"
|
||||
|
||||
# Substring
|
||||
# TODO: improve this syntax
|
||||
compile [%str |%start|] to: "\(%str as lua):sub(\(%start as lua), \(%start as lua))"
|
||||
compile [%str |%start - %stop|] to: "\(%str as lua):sub(\(%start as lua), \(%stop as lua))"
|
||||
|
||||
# Variable assignment operator, and += type versions
|
||||
compile [local %vars] to code:
|
||||
lua> ".."
|
||||
local locals = \%vars.type == "List" and \%vars.value or {\%vars};
|
||||
local identifiers = {};
|
||||
for i,x in ipairs(locals) do
|
||||
identifiers[i] = nomsu:tree_to_lua(x).expr;
|
||||
end
|
||||
return "local "..table.concat(identifiers, ", ");
|
||||
compile [set %var = %val] to code:
|
||||
lua> ".."
|
||||
if \%var.type == 'List' and \%val.type == 'List' then
|
||||
local lhs = {};
|
||||
for i,x in ipairs(\%var.value) do lhs[i] = nomsu:tree_to_lua(x).expr; end
|
||||
local rhs = {};
|
||||
for i,x in ipairs(\%val.value) do rhs[i] = nomsu:tree_to_lua(x).expr; end
|
||||
return table.concat(lhs, ", ").." = "..table.concat(rhs, ", ")..";";
|
||||
else
|
||||
return \(%var as lua).." = "..\(%val as lua)..";";
|
||||
end
|
||||
compile [%var += %val] to code: "\(%var as lua) = \(%var as lua) + \(%val as lua);"
|
||||
compile [%var -= %val] to code: "\(%var as lua) = \(%var as lua) - \(%val as lua);"
|
||||
compile [%var *= %val] to code: "\(%var as lua) = \(%var as lua) * \(%val as lua);"
|
||||
compile [%var /= %val] to code: "\(%var as lua) = \(%var as lua) / \(%val as lua);"
|
||||
compile [%var ^= %val] to code: "\(%var as lua) = \(%var as lua) ^ \(%val as lua);"
|
||||
compile [%var and= %val] to code: "\(%var as lua) = \(%var as lua) and\(%val as lua);"
|
||||
compile [%var or= %val] to code: "\(%var as lua) = \(%var as lua) or \(%val as lua);"
|
||||
compile [%var join= %val] to code: "\(%var as lua) = \(%var as lua) .. \(%val as lua);"
|
||||
compile [wrap %var around %val] to code: "\(%var as lua) = \(%var as lua) % \(%val as lua);"
|
||||
|
||||
# Math Operators
|
||||
compile [%x + %y] to: "(\(%x as lua) + \(%y as lua))"
|
||||
compile [%x - %y] to: "(\(%x as lua) - \(%y as lua))"
|
||||
compile [%x * %y] to: "(\(%x as lua) * \(%y as lua))"
|
||||
compile [%x / %y] to: "(\(%x as lua) / \(%y as lua))"
|
||||
compile [%x ^ %y] to: "(\(%x as lua) ^ \(%y as lua))"
|
||||
compile [%x wrapped around %y, %x mod %y] to: "(\(%x as lua) % \(%y as lua))"
|
||||
|
||||
# Comparison Operators
|
||||
immediately:
|
||||
compile [%x < %y] to: "(\(%x as lua) < \(%y as lua))"
|
||||
compile [%x > %y] to: "(\(%x as lua) > \(%y as lua))"
|
||||
compile [%x <= %y] to: "(\(%x as lua) <= \(%y as lua))"
|
||||
@ -110,6 +34,46 @@ compile [%a isn't %b, %a is not %b, %a not= %b, %a != %b] to:
|
||||
# For strict identity checking, use (%x's id) is (%y's id)
|
||||
compile [%'s id, id of %] to: "nomsu.ids[\(% as lua)]"
|
||||
|
||||
# Variable assignment operator, and += type versions
|
||||
immediately:
|
||||
compile [local %vars] to code:
|
||||
lua> ".."
|
||||
local locals = \%vars.type == "List" and \%vars.value or {\%vars};
|
||||
local identifiers = {};
|
||||
for i,x in ipairs(locals) do
|
||||
identifiers[i] = nomsu:tree_to_lua(x).expr;
|
||||
end
|
||||
return "local "..table.concat(identifiers, ", ");
|
||||
compile [set %var = %val] to code: "\(%var as lua) = \(%val as lua);"
|
||||
compile [set %assignments] to code:
|
||||
assume ((%assignments' "type") is "Dict") or barf "Expected Dict, but got \(%assignments' "type")"
|
||||
lua> ".."
|
||||
local lhs, rhs = {}, {};
|
||||
for i,entry in ipairs(\%assignments.value) do
|
||||
lhs[i] = nomsu:tree_to_lua(entry.dict_key).expr;
|
||||
rhs[i] = nomsu:tree_to_lua(entry.dict_value).expr;
|
||||
end
|
||||
return table.concat(lhs, ", ").." = "..table.concat(rhs, ", ")..";";
|
||||
|
||||
# Update assignment operators
|
||||
compile [%var += %val] to code: "\(%var as lua) = \(%var as lua) + \(%val as lua);"
|
||||
compile [%var -= %val] to code: "\(%var as lua) = \(%var as lua) - \(%val as lua);"
|
||||
compile [%var *= %val] to code: "\(%var as lua) = \(%var as lua) * \(%val as lua);"
|
||||
compile [%var /= %val] to code: "\(%var as lua) = \(%var as lua) / \(%val as lua);"
|
||||
compile [%var ^= %val] to code: "\(%var as lua) = \(%var as lua) ^ \(%val as lua);"
|
||||
compile [%var and= %val] to code: "\(%var as lua) = \(%var as lua) and\(%val as lua);"
|
||||
compile [%var or= %val] to code: "\(%var as lua) = \(%var as lua) or \(%val as lua);"
|
||||
compile [%var join= %val] to code: "\(%var as lua) = \(%var as lua) .. \(%val as lua);"
|
||||
compile [wrap %var around %val] to code: "\(%var as lua) = \(%var as lua) % \(%val as lua);"
|
||||
|
||||
# Math Operators
|
||||
compile [%x + %y] to: "(\(%x as lua) + \(%y as lua))"
|
||||
compile [%x - %y] to: "(\(%x as lua) - \(%y as lua))"
|
||||
compile [%x * %y] to: "(\(%x as lua) * \(%y as lua))"
|
||||
compile [%x / %y] to: "(\(%x as lua) / \(%y as lua))"
|
||||
compile [%x ^ %y] to: "(\(%x as lua) ^ \(%y as lua))"
|
||||
compile [%x wrapped around %y, %x mod %y] to: "(\(%x as lua) % \(%y as lua))"
|
||||
|
||||
# 3-part chained comparisons
|
||||
# (uses a lambda to avoid re-evaluating middle value, while still being an expression)
|
||||
parse [%x < %y < %z] as: =lua "(function(x,y,z) return x < y and y < z; end)(\%x,\%y,\%z)"
|
||||
|
59
lib/text.nom
59
lib/text.nom
@ -1,7 +1,62 @@
|
||||
#..
|
||||
This file contains some definitions of text escape sequences, including ANSI console
|
||||
color codes.
|
||||
|
||||
use "lib/metaprogramming.nom"
|
||||
|
||||
# Text functions
|
||||
action [join %strs with %glue]:
|
||||
lua> ".."
|
||||
local str_bits = {}
|
||||
for i,bit in ipairs(\%strs) do str_bits[i] = nomsu:stringify(bit) end
|
||||
return table.concat(str_bits, \%glue)
|
||||
parse [join %strs] as: join %strs with ""
|
||||
|
||||
compile [capitalize %str, %str capitalized] to:
|
||||
"(\(%str as lua)):gsub('%l', string.upper, 1)"
|
||||
|
||||
compile [%str with %patt replaced with %sub, %str s/%patt/%sub] to:
|
||||
"((\(%str as lua)):gsub(\(%patt as lua), \(%sub as lua)))"
|
||||
compile [%str with %patt replaced with %sub %n times, %str s/%patt/%sub/%n] to:
|
||||
"((\(%str as lua)):gsub(\(%patt as lua), \(%sub as lua), \(%n as lua)))"
|
||||
|
||||
# Substring
|
||||
# TODO: improve this syntax
|
||||
compile [%str |%start|] to: "\(%str as lua):sub(\(%start as lua), \(%start as lua))"
|
||||
compile [%str |%start - %stop|] to: "\(%str as lua):sub(\(%start as lua), \(%stop as lua))"
|
||||
|
||||
# Text literals
|
||||
lua do> ".."
|
||||
local escapes = {
|
||||
nl="\\\\n", newline="\\\\n", tab="\\\\t", bell="\\\\a", cr="\\\\r", ["carriage return"]="\\\\r",
|
||||
backspace="\\\\b", ["form feed"]="\\\\f", formfeed="\\\\f", ["vertical tab"]="\\\\v",
|
||||
};
|
||||
local colors = {
|
||||
["reset color"]="\\\\27[0m", bright="\\\\27[1m", dim="\\\\27[2m", underscore="\\\\27[4m",
|
||||
blink="\\\\27[5m", inverse="\\\\27[7m", hidden="\\\\27[8m",
|
||||
|
||||
black="\\\\27[30m", red="\\\\27[31m", green="\\\\27[32m", yellow="\\\\27[33m", blue="\\\\27[34m",
|
||||
magenta="\\\\27[35m", cyan="\\\\27[36m", white="\\\\27[37m",
|
||||
|
||||
["on black"]="\\\\27[40m", ["on red"]="\\\\27[41m", ["on green"]="\\\\27[42m", ["on yellow"]="\\\\27[43m",
|
||||
["on blue"]="\\\\27[44m", ["on magenta"]="\\\\27[45m", ["on cyan"]="\\\\27[46m", ["on white"]="\\\\27[47m",
|
||||
};
|
||||
for name, e in pairs(escapes) do
|
||||
local lua = "'"..e.."'";
|
||||
nomsu:define_compile_action(name, \(__line_no__), function() return {expr=str}; end, \(__src__ 1));
|
||||
end
|
||||
for name, c in pairs(colors) do
|
||||
local color = "'"..c.."'";
|
||||
local reset = "'"..colors["reset color"].."'";
|
||||
nomsu:define_compile_action(name, \(__line_no__), function() return {expr=color}; end, \(__src__ 1));
|
||||
nomsu:define_compile_action(name.." %", \(__line_no__), function(nomsu, _)
|
||||
return {expr=color..".."..nomsu:tree_to_lua(_).expr..".."..reset};
|
||||
end, \(__src__ 1));
|
||||
end
|
||||
|
||||
#..
|
||||
use "lib/control_flow.nom"
|
||||
use "lib/collections.nom"
|
||||
|
||||
local [%ansi, %colors]
|
||||
set %ansi = {..}
|
||||
nl="\\n", newline="\\n", tab="\\t", bell="\\a", cr="\\r", "carriage return"="\\r"
|
||||
@ -30,3 +85,5 @@ for %name=%str in %colors:
|
||||
nomsu:define_compile_action(\%name.." %", \(__line_no__), function(nomsu, _)
|
||||
return {expr=e..".."..nomsu:tree_to_lua(_).expr..".."..reset};
|
||||
end, \(__src__ 1));
|
||||
|
||||
# FIXME
|
||||
|
@ -1,3 +1,7 @@
|
||||
#..
|
||||
This file contains a set of definitions that bring some familiar language features
|
||||
from other languages into nomsu (e.g. "==" and "continue")
|
||||
|
||||
use "lib/core.nom"
|
||||
|
||||
parse [%a == %b] as: %a is %b
|
||||
@ -20,3 +24,4 @@ parse [let %assignments in %action] as: with %assignemnts %action
|
||||
parse [error!, panic!, fail!, abort!] as: barf!
|
||||
parse [error %, panic %, fail %, abort %] as: barf %
|
||||
parse [assert %condition %message] as: assume %condition or barf %message
|
||||
parse [%cond ? %if_true %if_false] as: %if_true if %cond else %if_false
|
||||
|
@ -1,95 +0,0 @@
|
||||
use "lib/metaprogramming.nom"
|
||||
|
||||
# Error functions
|
||||
action [barf!]:
|
||||
nomsu "error" []
|
||||
action [barf %msg]:
|
||||
nomsu "error"[%msg]
|
||||
compile [assume %condition or barf %msg] to code: ".."
|
||||
if not (\(%condition as lua)) then
|
||||
nomsu:error(\(%msg as lua));
|
||||
end
|
||||
|
||||
parse [assume %condition] as: assume %condition or barf (nil)
|
||||
|
||||
# Text functions
|
||||
action [join %strs with glue %glue]:
|
||||
lua do> ".."
|
||||
local str_bits = {}
|
||||
for i,bit in ipairs(\%strs) do str_bits[i] = nomsu:stringify(bit) end
|
||||
return table.concat(str_bits, \%glue)
|
||||
parse [join %strs] as: join %strs with glue ""
|
||||
|
||||
compile [capitalize %str, %str capitalized] to:
|
||||
"(\(%str as lua)):gsub('%l', string.upper, 1)"
|
||||
|
||||
compile [%str with %patt replaced with %sub, %str s/%patt/%sub] to:
|
||||
"((\(%str as lua)):gsub(\(%patt as lua), \(%sub as lua)))"
|
||||
compile [%str with %patt replaced with %sub %n times, %str s/%patt/%sub/%n] to:
|
||||
"((\(%str as lua)):gsub(\(%patt as lua), \(%sub as lua), \(%n as lua)))"
|
||||
|
||||
# Number ranges
|
||||
compile [%start to %stop by %step, %start to %stop via %step] to: ".."
|
||||
nomsu.utils.range(\(%start as lua), \(%stop as lua), \(%step as lua))
|
||||
|
||||
parse [%start to %stop] as: %start to %stop by 1
|
||||
|
||||
# Random functions
|
||||
lua> ".." # Seed
|
||||
math.randomseed(os.time());
|
||||
for i=1,20 do; math.random(); end;
|
||||
compile [random number, random, rand] to: "math.random()"
|
||||
compile [random int %n, random integer %n, randint %n] to: "math.random(\(%n as lua))"
|
||||
compile [random from %low to %high, random number from %low to %high, rand %low %high] to:
|
||||
"math.random(\(%low as lua), \(%high as lua))"
|
||||
action [random choice from %elements, random choice %elements, random %elements]:
|
||||
=lua "\%elements[math.random(#\%elements)]"
|
||||
|
||||
# Math functions
|
||||
compile [% as number] to: "tonumber(\(% as lua))"
|
||||
compile [abs %, absolute value of %, | % |] to: "math.abs(\(% as lua))"
|
||||
compile [sqrt %, square root of %] to: "math.sqrt(\(% as lua))"
|
||||
compile [sin %, sine %] to: "math.sin(\(% as lua))"
|
||||
compile [cos %, cosine %] to: "math.cos(\(% as lua))"
|
||||
compile [tan %, tangent %] to: "math.tan(\(% as lua))"
|
||||
compile [asin %, arc sine %] to: "math.asin(\(% as lua))"
|
||||
compile [acos %, arc cosine %] to: "math.acos(\(% as lua))"
|
||||
compile [atan %, arc tangent %] to: "math.atan(\(% as lua))"
|
||||
compile [atan2 %y %x] to: "math.atan2(\(%y as lua), \(%x as lua))"
|
||||
compile [sinh %, hyperbolic sine %] to: "math.sinh(\(% as lua))"
|
||||
compile [cosh %, hyperbolic cosine %] to: "math.cosh(\(% as lua))"
|
||||
compile [tanh %, hyperbolic tangent %] to: "math.tanh(\(% as lua))"
|
||||
compile [ceil %, ceiling %] to: "math.ceil(\(% as lua))"
|
||||
compile [exp %, e^ %] to: "math.exp(\(% as lua))"
|
||||
compile [log %, ln %, natural log %] to: "math.log(\(% as lua))"
|
||||
compile [log % base %base] to: "math.log(\(% as lua), \(%base as lua))"
|
||||
compile [floor %] to: "math.floor(\(% as lua))"
|
||||
compile [round %, % rounded] to: "math.floor(\(% as lua) + .5)"
|
||||
action [%n rounded to the nearest %rounder]:
|
||||
=lua "(\%rounder)*math.floor((\%n / \%rounder) + .5)"
|
||||
compile [tau] to: "(2*math.pi)"
|
||||
compile [pi] to: "math.pi"
|
||||
compile [e] to: "math.e"
|
||||
compile [golden ratio, phi] to: "((1 + math.sqrt(5)) / 2)"
|
||||
|
||||
# Common utility functions
|
||||
action [avg of %items, average of %items]:
|
||||
=lua "(nomsu.utils.sum(\%items)/#\%items)"
|
||||
compile [min of %items, smallest of %items, lowest of %items] to:
|
||||
"nomsu.utils.min(\(%items as lua))"
|
||||
compile [max of %items, biggest of %items, largest of %items, highest of %items] to:
|
||||
"nomsu.utils.max(\(%items as lua))"
|
||||
compile [min of %items by %value_expr] to: ".."
|
||||
nomsu.utils.min(\(%items as lua), function(\(\% as lua))
|
||||
return \(%value_expr as lua)
|
||||
end)
|
||||
compile [max of %items by %value_expr] to: ".."
|
||||
nomsu.utils.max(\(%items as lua), function(\(\% as lua))
|
||||
return \(%value_expr as lua)
|
||||
end)
|
||||
compile [sort %items] to: "table.sort(\(%items as lua))"
|
||||
compile [sort %items by %key_expr] to: ".."
|
||||
nomsu.utils.sort(\(%items as lua), function(\(\% as lua))
|
||||
return \(%key_expr as lua);
|
||||
end)
|
||||
|
@ -1,63 +0,0 @@
|
||||
use "lib/metaprogramming.nom"
|
||||
use "lib/utils.nom"
|
||||
use "lib/control_flow.nom"
|
||||
use "lib/operators.nom"
|
||||
use "lib/collections.nom"
|
||||
|
||||
|
||||
compile [say %str] to:
|
||||
"nomsu:writeln(\(%str as lua))" if ((%str's "type") is "Text")
|
||||
..else "nomsu:writeln(nomsu:stringify(\(%str as lua)))"
|
||||
|
||||
compile [do %action] to code:
|
||||
(%action as lua statements) if ((%action's "type") is "Block")
|
||||
..else "(\(%action as lua))(nomsu);"
|
||||
|
||||
# With statement
|
||||
compile [with %assignments %action] to code:
|
||||
set %data = []
|
||||
for %i = %assignment in (%assignments' "value"):
|
||||
set %tokens = (%assignment's "value")
|
||||
set %var = (%tokens -> 1)
|
||||
set %eq = (%tokens -> 2)
|
||||
assume (=lua "\%eq and \%eq.type == 'Word' and \%eq.value == '='") or barf ".."
|
||||
Invalid format for 'with' statement. List entries must have the form %var = (value)
|
||||
set %value = (%tokens -> 3)
|
||||
add {i=%i, var=%var, value=%value} to %data
|
||||
set %setup = (..)
|
||||
join (..)
|
||||
"local old_value\(%->"i") = \((%->"var") as lua); \((%->"var") as lua) = \((%->"value") as lua);"
|
||||
..for all %data
|
||||
..with glue "\n "
|
||||
return ".."
|
||||
do
|
||||
\%setup
|
||||
local fell_through = false;
|
||||
local ok, ret1, ret2 = pcall(function(nomsu)
|
||||
\(%action as lua statements);
|
||||
fell_through = true;
|
||||
end, nomsu);
|
||||
\(join ("\((%->"var") as lua) = old_value\(%->"i");" for all %data) with glue "\n ")
|
||||
if not ok then nomsu:error(ret1); end
|
||||
if not fell_through then
|
||||
return ret1, ret2;
|
||||
end
|
||||
end
|
||||
parse [with %thing = %value %action] as: with [%thing = %value] %action
|
||||
|
||||
# Any/all/none
|
||||
compile [all of %items, all %items] to:
|
||||
"(\(join ((% as lua) for all (%items' "value")) with glue " and "))"
|
||||
..if ((%items' "type") is "List") else "nomsu.utils.all(\(%items as lua))"
|
||||
parse [not all of %items, not all %items] as: not (all of %items)
|
||||
compile [any of %items, any %items] to:
|
||||
"(\(join ((% as lua) for all (%items' "value")) with glue " or "))"
|
||||
..if ((%items' "type") is "List") else "nomsu.utils.any(\(%items as lua))"
|
||||
parse [none of %items, none %items] as: not (any of %items)
|
||||
|
||||
compile [sum of %items, sum %items] to:
|
||||
"(\(join ((% as lua) for all (%items' "value")) with glue " + "))"
|
||||
..if ((%items' "type") is "List") else "nomsu.utils.sum(\(%items as lua))"
|
||||
compile [product of %items, product %items] to:
|
||||
"(\(join ((% as lua) for all (%items' "value")) with glue " * "))"
|
||||
..if ((%items' "type") is "List") else "nomsu.utils.product(\(%items as lua))"
|
Loading…
Reference in New Issue
Block a user