Added and fixed up "when"
This commit is contained in:
parent
77d37aaf0f
commit
e073b23fbf
82
lib/core.nom
82
lib/core.nom
@ -167,22 +167,27 @@ lua block ".."
|
|||||||
| compiler:error("Assignment operation has the wrong type for the right hand side. "
|
| compiler:error("Assignment operation has the wrong type for the right hand side. "
|
||||||
| .."Expected Thunk, but got: "..vars.rhs.type.."\\nMaybe you used '=' instead of '=:'?")
|
| .."Expected Thunk, but got: "..vars.rhs.type.."\\nMaybe you used '=' instead of '=:'?")
|
||||||
| end
|
| end
|
||||||
|
| if #vars.rhs.value.value == 1 then
|
||||||
|
| return callback(compiler:tree_to_lua(vars.var, "Expression"),
|
||||||
|
| compiler:tree_to_lua(vars.rhs.value.value[1].value, "Expression")), true
|
||||||
|
| else
|
||||||
| local ret = "do\\n local ret"
|
| local ret = "do\\n local ret"
|
||||||
| ret = ret .. "\\n "..compiler:tree_to_lua(vars.rhs.value, "Statement")
|
| ret = ret .. "\\n "..compiler:tree_to_lua(vars.rhs.value, "Statement")
|
||||||
| ret = ret .. "\\n "..callback(compiler:tree_to_lua(vars.var, "Expression"))
|
| ret = ret .. "\\n "..callback(compiler:tree_to_lua(vars.var, "Expression"), "ret")
|
||||||
| return (ret.."\\nend"), true
|
| return (ret.."\\nend"), true
|
||||||
| end
|
| end
|
||||||
| end
|
| end
|
||||||
|compiler:defmacro("%var = %rhs", helper(function(var) return var.." = ret" end))
|
|end
|
||||||
|compiler:defmacro("%var += %rhs", helper(function(var) return var.." = "..var.." + ret" end))
|
|compiler:defmacro("%var = %rhs", helper(function(var,result) return var.." = "..result end))
|
||||||
|compiler:defmacro("%var -= %rhs", helper(function(var) return var.." = "..var.." - ret" end))
|
|compiler:defmacro("%var += %rhs", helper(function(var,result) return var.." = "..var.." + "..result end))
|
||||||
|compiler:defmacro("%var *= %rhs", helper(function(var) return var.." = "..var.." * ret" end))
|
|compiler:defmacro("%var -= %rhs", helper(function(var,result) return var.." = "..var.." - "..result end))
|
||||||
|compiler:defmacro("%var /= %rhs", helper(function(var) return var.." = "..var.." / ret" end))
|
|compiler:defmacro("%var *= %rhs", helper(function(var,result) return var.." = "..var.." * "..result end))
|
||||||
|compiler:defmacro("%var ^= %rhs", helper(function(var) return var.." = "..var.." ^ ret" end))
|
|compiler:defmacro("%var /= %rhs", helper(function(var,result) return var.." = "..var.." / "..result end))
|
||||||
|compiler:defmacro("%var and= %rhs", helper(function(var) return var.." = "..var.." and ret" end))
|
|compiler:defmacro("%var ^= %rhs", helper(function(var,result) return var.." = "..var.." ^ "..result end))
|
||||||
|compiler:defmacro("%var or= %rhs", helper(function(var) return var.." = "..var.." or ret" end))
|
|compiler:defmacro("%var and= %rhs", helper(function(var,result) return var.." = "..var.." and "..result end))
|
||||||
|compiler:defmacro("%var concat= %rhs", helper(function(var) return var.." = "..var.." .. ret" end))
|
|compiler:defmacro("%var or= %rhs", helper(function(var,result) return var.." = "..var.." or "..result end))
|
||||||
|compiler:defmacro("%var mod= %rhs", helper(function(var) return var.." = "..var.." % ret" end))
|
|compiler:defmacro("%var concat= %rhs", helper(function(var,result) return var.." = "..var.." .. "..result end))
|
||||||
|
|compiler:defmacro("%var mod= %rhs", helper(function(var,result) return var.." = "..var.." % "..result end))
|
||||||
|
|
||||||
# Operators
|
# Operators
|
||||||
macro [true, yes] =: "true"
|
macro [true, yes] =: "true"
|
||||||
@ -279,11 +284,6 @@ macro [%if_expr if %condition else %else_expr] =:
|
|||||||
| end
|
| end
|
||||||
|end)(compiler, vars)
|
|end)(compiler, vars)
|
||||||
|
|
||||||
# Switch statement/multi-branch if
|
|
||||||
macro block [when %body] =:
|
|
||||||
%result =: ""
|
|
||||||
%result
|
|
||||||
|
|
||||||
|
|
||||||
# Loop control flow
|
# Loop control flow
|
||||||
macro block [break] =: "break"
|
macro block [break] =: "break"
|
||||||
@ -350,6 +350,45 @@ macro block [for all %iterable %body] =:
|
|||||||
|end
|
|end
|
||||||
|vars.it = old_loopval
|
|vars.it = old_loopval
|
||||||
|
|
||||||
|
# Switch statement/multi-branch if
|
||||||
|
macro block [when %body] =:
|
||||||
|
%result =: ""
|
||||||
|
for %statement in (lua expr "vars.body.value.value"):
|
||||||
|
%func-call =: lua expr "vars.statement.value"
|
||||||
|
if ((lua expr "vars['func-call'].type") != "FunctionCall"):
|
||||||
|
error "Invalid format for 'when' statement"
|
||||||
|
%tokens =: lua expr "vars['func-call'].value"
|
||||||
|
%star =: lua expr "vars.tokens[1]"
|
||||||
|
if (((lua expr "vars.star.type") != "Word") or ((lua expr "vars.star.value") != "*")):
|
||||||
|
error "Invalid format for 'when' statement"
|
||||||
|
%thunk =: lua expr "vars.tokens[#vars.tokens]"
|
||||||
|
if ((lua expr "vars.thunk.type") != "Thunk"):
|
||||||
|
error "Invalid format for 'when' statement"
|
||||||
|
%condition-bits =: []
|
||||||
|
for %i in (2 up to (lua expr "#vars.tokens")):
|
||||||
|
lua block "table.insert(vars['condition-bits'], vars.tokens[vars.i])"
|
||||||
|
%condition =: dict [..]
|
||||||
|
["type",lua expr "vars['func-call'].type"]
|
||||||
|
["src",lua expr "vars['func-call'].src"]
|
||||||
|
["value", %condition-bits]
|
||||||
|
if ((lua expr "#vars.condition.value") == 0):
|
||||||
|
%result concat=: ".."
|
||||||
|
|
|
||||||
|
|do
|
||||||
|
| local ret
|
||||||
|
| \(lua expr "vars.thunk.value") as lua block\
|
||||||
|
| return ret
|
||||||
|
|end
|
||||||
|
..else:
|
||||||
|
%result concat=: ".."
|
||||||
|
|
|
||||||
|
|if \%condition as lua expr\ then
|
||||||
|
| local ret
|
||||||
|
| \(lua expr "vars.thunk.value") as lua block\
|
||||||
|
| return ret
|
||||||
|
|end
|
||||||
|
%result
|
||||||
|
|
||||||
# List Comprehension
|
# List Comprehension
|
||||||
# TODO: maybe make this lazy, or a lazy version?
|
# TODO: maybe make this lazy, or a lazy version?
|
||||||
macro [%expression for %var in %iterable] =:
|
macro [%expression for %var in %iterable] =:
|
||||||
@ -452,6 +491,17 @@ macro [..]
|
|||||||
%index th in %list, %index in %list, %list -> %index
|
%index th in %list, %index in %list, %list -> %index
|
||||||
..=:
|
..=:
|
||||||
".."|\%list as lua expr\[\%index as lua expr\]
|
".."|\%list as lua expr\[\%index as lua expr\]
|
||||||
|
".."|\%list as lua expr\[\%index as lua expr\]
|
||||||
|
macro [first in %list] =:
|
||||||
|
".."|\%list as lua expr\[1]
|
||||||
|
|
||||||
|
macro [..]
|
||||||
|
%index st to last in %list, %index nd to last in %list, %index rd to last in %list
|
||||||
|
%index th to last in %list
|
||||||
|
..=:
|
||||||
|
".."|compiler.utils.nth_to_last(\%list as lua expr\, \%index as lua expr\)
|
||||||
|
macro [last in %list] =:
|
||||||
|
".."|compiler.utils.nth_to_last(\%list as lua expr\, 1)
|
||||||
|
|
||||||
macro block [..]
|
macro block [..]
|
||||||
%list's %index = %new_value, %index st in %list = %new_value, %index nd in %list = %new_value
|
%list's %index = %new_value, %index st in %list = %new_value, %index nd in %list = %new_value
|
||||||
|
401
nomsu.lua
401
nomsu.lua
@ -5,133 +5,6 @@ local INDENT = " "
|
|||||||
lpeg.setmaxstack(10000)
|
lpeg.setmaxstack(10000)
|
||||||
local P, V, S, Cg, C, Cp, B, Cmt
|
local P, V, S, Cg, C, Cp, B, Cmt
|
||||||
P, V, S, Cg, C, Cp, B, Cmt = lpeg.P, lpeg.V, lpeg.S, lpeg.Cg, lpeg.C, lpeg.Cp, lpeg.B, lpeg.Cmt
|
P, V, S, Cg, C, Cp, B, Cmt = lpeg.P, lpeg.V, lpeg.S, lpeg.Cg, lpeg.C, lpeg.Cp, lpeg.B, lpeg.Cmt
|
||||||
local get_line_indentation
|
|
||||||
get_line_indentation = function(line)
|
|
||||||
local indent_amounts = {
|
|
||||||
[" "] = 1,
|
|
||||||
["\t"] = 4
|
|
||||||
}
|
|
||||||
do
|
|
||||||
local sum = 0
|
|
||||||
local leading_space = line:match("[\t ]*")
|
|
||||||
for c in leading_space:gmatch("[\t ]") do
|
|
||||||
sum = sum + indent_amounts[c]
|
|
||||||
end
|
|
||||||
return sum
|
|
||||||
end
|
|
||||||
end
|
|
||||||
local make_parser
|
|
||||||
make_parser = function(lingo, extra_definitions)
|
|
||||||
local indent_stack = {
|
|
||||||
0
|
|
||||||
}
|
|
||||||
local push
|
|
||||||
push = function(n)
|
|
||||||
return table.insert(indent_stack, n)
|
|
||||||
end
|
|
||||||
local pop
|
|
||||||
pop = function()
|
|
||||||
return table.remove(indent_stack)
|
|
||||||
end
|
|
||||||
local check_indent
|
|
||||||
check_indent = function(subject, end_pos, spaces)
|
|
||||||
local num_spaces = get_line_indentation(spaces)
|
|
||||||
if num_spaces <= indent_stack[#indent_stack] then
|
|
||||||
return nil
|
|
||||||
end
|
|
||||||
push(num_spaces)
|
|
||||||
return end_pos
|
|
||||||
end
|
|
||||||
local check_dedent
|
|
||||||
check_dedent = function(subject, end_pos, spaces)
|
|
||||||
local num_spaces = get_line_indentation(spaces)
|
|
||||||
if num_spaces >= indent_stack[#indent_stack] then
|
|
||||||
return nil
|
|
||||||
end
|
|
||||||
pop()
|
|
||||||
return end_pos
|
|
||||||
end
|
|
||||||
local check_nodent
|
|
||||||
check_nodent = function(subject, end_pos, spaces)
|
|
||||||
local num_spaces = get_line_indentation(spaces)
|
|
||||||
if num_spaces ~= indent_stack[#indent_stack] then
|
|
||||||
return nil
|
|
||||||
end
|
|
||||||
return end_pos
|
|
||||||
end
|
|
||||||
local wordchar = P(1) - S(' \t\n\r%#:;,.{}[]()"\\')
|
|
||||||
local nl = P("\n")
|
|
||||||
local whitespace = S(" \t") ^ 1
|
|
||||||
local blank_line = whitespace ^ -1 * nl
|
|
||||||
local line_comment = re.compile([=[ "#" [^%nl]* ]=], {
|
|
||||||
nl = nl
|
|
||||||
})
|
|
||||||
local block_comment = re.compile([=[ "#.." (!%nl .)* (%indent (!%dedent %nl [^%nl]*)*)
|
|
||||||
]=], {
|
|
||||||
nl = nl,
|
|
||||||
whitespace = whitespace,
|
|
||||||
indent = #(nl * blank_line ^ 0 * Cmt(S(" \t") ^ 0, check_indent)),
|
|
||||||
dedent = #(nl * blank_line ^ 0 * Cmt(S(" \t") ^ 0, check_dedent)),
|
|
||||||
new_line = nl * blank_line ^ 0 * Cmt(S(" \t") ^ 0, check_nodent)
|
|
||||||
})
|
|
||||||
blank_line = ((Cmt(whitespace ^ -1, check_nodent) * (block_comment + line_comment)) ^ -1 + whitespace ^ -1) * nl
|
|
||||||
local defs = {
|
|
||||||
wordchar = wordchar,
|
|
||||||
nl = nl,
|
|
||||||
ws = whitespace,
|
|
||||||
blank_line = blank_line,
|
|
||||||
block_comment = block_comment,
|
|
||||||
line_comment = line_comment,
|
|
||||||
eol = #nl + (P("") - P(1)),
|
|
||||||
word_boundary = whitespace ^ -1 + B(P("..")) + B(S("\";)]")) + #S("\":([") + #((whitespace + nl) ^ 0 * P("..")),
|
|
||||||
indent = #(nl * blank_line ^ 0 * Cmt(whitespace ^ -1, check_indent)),
|
|
||||||
dedent = #(nl * blank_line ^ 0 * Cmt(whitespace ^ -1, check_dedent)),
|
|
||||||
new_line = nl * blank_line ^ 0 * Cmt(whitespace ^ -1, check_nodent),
|
|
||||||
error_handler = function(src, pos, errors)
|
|
||||||
local line_no = 1
|
|
||||||
for _ in src:sub(1, -#errors):gmatch("\n") do
|
|
||||||
line_no = line_no + 1
|
|
||||||
end
|
|
||||||
local err_pos = #src - #errors + 1
|
|
||||||
if errors:sub(1, 1) == "\n" then
|
|
||||||
err_pos = err_pos + #errors:match("[ \t]*", 2)
|
|
||||||
end
|
|
||||||
local start_of_err_line = err_pos
|
|
||||||
while src:sub(start_of_err_line, start_of_err_line) ~= "\n" do
|
|
||||||
start_of_err_line = start_of_err_line - 1
|
|
||||||
end
|
|
||||||
local start_of_prev_line = start_of_err_line - 1
|
|
||||||
while src:sub(start_of_prev_line, start_of_prev_line) ~= "\n" do
|
|
||||||
start_of_prev_line = start_of_prev_line - 1
|
|
||||||
end
|
|
||||||
local prev_line, err_line, next_line = src:match("([^\n]*)\n([^\n]*)\n([^\n]*)", start_of_prev_line + 1)
|
|
||||||
local pointer = ("-"):rep(err_pos - start_of_err_line + 0) .. "^"
|
|
||||||
return error("\nParse error on line " .. tostring(line_no) .. ":\n\n" .. tostring(prev_line) .. "\n" .. tostring(err_line) .. "\n" .. tostring(pointer) .. "\n" .. tostring(next_line) .. "\n")
|
|
||||||
end
|
|
||||||
}
|
|
||||||
if extra_definitions then
|
|
||||||
for k, v in pairs(extra_definitions) do
|
|
||||||
defs[k] = v
|
|
||||||
end
|
|
||||||
end
|
|
||||||
setmetatable(defs, {
|
|
||||||
__index = function(t, key)
|
|
||||||
local fn
|
|
||||||
fn = function(src, value, errors)
|
|
||||||
local token = {
|
|
||||||
type = key,
|
|
||||||
src = src,
|
|
||||||
value = value,
|
|
||||||
errors = errors
|
|
||||||
}
|
|
||||||
return token
|
|
||||||
end
|
|
||||||
t[key] = fn
|
|
||||||
return fn
|
|
||||||
end
|
|
||||||
})
|
|
||||||
return re.compile(lingo, defs)
|
|
||||||
end
|
|
||||||
local NomsuCompiler
|
local NomsuCompiler
|
||||||
do
|
do
|
||||||
local _class_0
|
local _class_0
|
||||||
@ -204,32 +77,45 @@ do
|
|||||||
end
|
end
|
||||||
end,
|
end,
|
||||||
get_invocations_from_definition = function(self, def, vars)
|
get_invocations_from_definition = function(self, def, vars)
|
||||||
if def.type == "String" or def.type == "List" then
|
if def.type == "String" then
|
||||||
return self:tree_to_value(def, vars)
|
return self:tree_to_value(def, vars)
|
||||||
end
|
end
|
||||||
if def.type ~= "Thunk" then
|
if def.type ~= "List" then
|
||||||
self:error("Trying to get invocations from " .. tostring(def.type) .. ", but expected Thunk.")
|
error("DEF IS: " .. tostring(utils.repr(def)))
|
||||||
|
self:error("Trying to get invocations from " .. tostring(def.type) .. ", but expected List or String.")
|
||||||
end
|
end
|
||||||
local invocations = { }
|
local invocations = { }
|
||||||
local _list_0 = def.value.value
|
local _list_0 = def.value
|
||||||
for _index_0 = 1, #_list_0 do
|
for _index_0 = 1, #_list_0 do
|
||||||
local statement = _list_0[_index_0]
|
local _continue_0 = false
|
||||||
if statement.value.type ~= "FunctionCall" then
|
repeat
|
||||||
self:error("Invalid statement type: " .. tostring(statement.value.type) .. ", expected FunctionCall")
|
local item = _list_0[_index_0]
|
||||||
|
if item.type == "String" then
|
||||||
|
table.insert(invocations, self:tree_to_value(item, vars))
|
||||||
|
_continue_0 = true
|
||||||
|
break
|
||||||
|
end
|
||||||
|
if item.type ~= "FunctionCall" then
|
||||||
|
self:error("Invalid list item: " .. tostring(item.type) .. ", expected FunctionCall or String")
|
||||||
end
|
end
|
||||||
local name_bits = { }
|
local name_bits = { }
|
||||||
local _list_1 = statement.value.value
|
local _list_1 = item.value
|
||||||
for _index_1 = 1, #_list_1 do
|
for _index_1 = 1, #_list_1 do
|
||||||
local token = _list_1[_index_1]
|
local token = _list_1[_index_1]
|
||||||
if token.type == "Word" then
|
if token.type == "Word" then
|
||||||
table.insert(name_bits, token.value)
|
table.insert(name_bits, token.value)
|
||||||
elseif token.value.type == "Var" then
|
elseif token.type == "Var" then
|
||||||
table.insert(name_bits, token.value.src)
|
table.insert(name_bits, token.src)
|
||||||
else
|
else
|
||||||
self:error("Unexpected token type in definition: " .. tostring(token.value.type) .. " (expected Word or Var)")
|
self:error("Unexpected token type in definition: " .. tostring(token.type) .. " (expected Word or Var)")
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
table.insert(invocations, table.concat(name_bits, " "))
|
table.insert(invocations, table.concat(name_bits, " "))
|
||||||
|
_continue_0 = true
|
||||||
|
until true
|
||||||
|
if not _continue_0 then
|
||||||
|
break
|
||||||
|
end
|
||||||
end
|
end
|
||||||
return invocations
|
return invocations
|
||||||
end,
|
end,
|
||||||
@ -290,11 +176,11 @@ do
|
|||||||
self.defs[invocation] = fn_info
|
self.defs[invocation] = fn_info
|
||||||
end
|
end
|
||||||
end,
|
end,
|
||||||
run = function(self, text)
|
run = function(self, text, filename)
|
||||||
if self.debug then
|
if self.debug then
|
||||||
self:writeln("RUNNING TEXT:\n" .. tostring(text))
|
self:writeln("RUNNING TEXT:\n" .. tostring(text))
|
||||||
end
|
end
|
||||||
local code, retval = self:compile(text)
|
local code, retval = self:compile(text, filename)
|
||||||
if self.debug then
|
if self.debug then
|
||||||
self:writeln("\nGENERATED LUA CODE:\n" .. tostring(code))
|
self:writeln("\nGENERATED LUA CODE:\n" .. tostring(code))
|
||||||
self:writeln("\nPRODUCED RETURN VALUE:\n" .. tostring(retval))
|
self:writeln("\nPRODUCED RETURN VALUE:\n" .. tostring(retval))
|
||||||
@ -344,69 +230,173 @@ do
|
|||||||
end
|
end
|
||||||
return (lua_thunk())(self, { })
|
return (lua_thunk())(self, { })
|
||||||
end,
|
end,
|
||||||
parse = function(self, str)
|
parse = function(self, str, filename)
|
||||||
if self.debug then
|
if self.debug then
|
||||||
self:writeln("PARSING:\n" .. tostring(str))
|
self:writeln("PARSING:\n" .. tostring(str))
|
||||||
end
|
end
|
||||||
local lingo = [=[ file <- ({ {| %blank_line* {:body: block :} %blank_line* (errors)? |} }) -> File
|
local get_line_indentation
|
||||||
errors <- (({.+}) => error_handler)
|
get_line_indentation = function(line)
|
||||||
block <- ({ {| statement (%new_line statement)* |} }) -> Block
|
local indent_amounts = {
|
||||||
statement <- ({ (functioncall / expression) }) -> Statement
|
[" "] = 1,
|
||||||
one_liner <- ({ {|
|
["\t"] = 4
|
||||||
(({
|
}
|
||||||
(({ {|
|
do
|
||||||
(expression (%word_boundary fn_bit)+) / (word (%word_boundary fn_bit)*)
|
local sum = 0
|
||||||
|} }) -> FunctionCall)
|
local leading_space = line:match("[\t ]*")
|
||||||
/ (expression)
|
for c in leading_space:gmatch("[\t ]") do
|
||||||
}) -> Statement)
|
sum = sum + indent_amounts[c]
|
||||||
|} }) -> Block
|
end
|
||||||
|
return sum
|
||||||
|
end
|
||||||
|
end
|
||||||
|
local indent_stack = {
|
||||||
|
0
|
||||||
|
}
|
||||||
|
local check_indent
|
||||||
|
check_indent = function(subject, end_pos, spaces)
|
||||||
|
local num_spaces = get_line_indentation(spaces)
|
||||||
|
if num_spaces > indent_stack[#indent_stack] then
|
||||||
|
table.insert(indent_stack, num_spaces)
|
||||||
|
return end_pos
|
||||||
|
end
|
||||||
|
end
|
||||||
|
local check_dedent
|
||||||
|
check_dedent = function(subject, end_pos, spaces)
|
||||||
|
local num_spaces = get_line_indentation(spaces)
|
||||||
|
if num_spaces < indent_stack[#indent_stack] then
|
||||||
|
table.remove(indent_stack)
|
||||||
|
return end_pos
|
||||||
|
end
|
||||||
|
end
|
||||||
|
local check_nodent
|
||||||
|
check_nodent = function(subject, end_pos, spaces)
|
||||||
|
local num_spaces = get_line_indentation(spaces)
|
||||||
|
if num_spaces == indent_stack[#indent_stack] then
|
||||||
|
return end_pos
|
||||||
|
end
|
||||||
|
end
|
||||||
|
local lingo = [=[ file <- ({ {| shebang? {:body: block :} %nl* (({.+} ("" -> "Unexpected end of file")) => error)? |} }) -> File
|
||||||
|
|
||||||
functioncall <- ({ {| (expression %word_boundary fn_bits) / (word (%word_boundary fn_bits)?) |} }) -> FunctionCall
|
shebang <- "#!" [^%nl]* %nl
|
||||||
fn_bit <- (expression / word)
|
|
||||||
fn_bits <-
|
|
||||||
((".." %ws? %line_comment? (%indent %new_line indented_fn_bits %dedent) (%new_line ".." %ws? fn_bits)?)
|
|
||||||
/ (%new_line ".." fn_bit (%word_boundary fn_bits)?)
|
|
||||||
/ (fn_bit (%word_boundary fn_bits)?))
|
|
||||||
indented_fn_bits <-
|
|
||||||
fn_bit ((%new_line / %word_boundary) indented_fn_bits)?
|
|
||||||
|
|
||||||
thunk <-
|
block <- ({ {|
|
||||||
({ ":" %ws? %line_comment?
|
(ignored_line %nl)*
|
||||||
((%indent %new_line block ((%dedent (%new_line "..")?) / errors))
|
line_of_statements (nodent line_of_statements)*
|
||||||
/ (one_liner (%ws? (%new_line? ".."))?)) }) -> Thunk
|
(%nl ignored_line)* |} }) -> Block
|
||||||
|
inline_block <- ({ {| inline_line_of_statements |} }) -> Block
|
||||||
|
|
||||||
|
line_of_statements <- statement (%ws? ";" %ws? statement)*
|
||||||
|
inline_line_of_statements <- inline_statement (%ws? ";" %ws? inline_statement)*
|
||||||
|
|
||||||
|
statement <- ({ functioncall / expression }) -> Statement
|
||||||
|
inline_statement <- ({ inline_functioncall / expression }) -> Statement
|
||||||
|
|
||||||
|
expression <- (
|
||||||
|
longstring / string / number / variable / list / thunk / block_functioncall
|
||||||
|
/ ("(" %ws? (inline_thunk / inline_functioncall) %ws? ")"))
|
||||||
|
|
||||||
|
-- Function calls need at least one word in them
|
||||||
|
functioncall <- ({ {|
|
||||||
|
(expression (dotdot / tok_gap))* word ((dotdot / tok_gap) (expression / word))*
|
||||||
|
|} }) -> FunctionCall
|
||||||
|
inline_functioncall <- ({ {|
|
||||||
|
(expression tok_gap)* word (tok_gap (expression / word))*
|
||||||
|
|} }) -> FunctionCall
|
||||||
|
block_functioncall <- "(..)" indent functioncall (dedent / (({.+} ("" -> "Error while parsing block function call")) => error))
|
||||||
|
|
||||||
word <- ({ !number {%wordchar (!"'" %wordchar)*} }) -> Word
|
word <- ({ !number {%wordchar (!"'" %wordchar)*} }) -> Word
|
||||||
expression <- ({ (longstring / string / number / variable / list / thunk / subexpression) }) -> Expression
|
|
||||||
|
thunk <- ({ ":" ((indent block (dedent / (({.+} ("" -> "Error while parsing thunk")) => error)))
|
||||||
|
/ (%ws? inline_block)) }) -> Thunk
|
||||||
|
inline_thunk <- ({ ":" %ws? inline_block }) -> Thunk
|
||||||
|
|
||||||
string <- ({ (!longstring) '"' {(("\" [^%nl]) / [^"%nl])*} '"' }) -> String
|
string <- ({ (!longstring) '"' {(("\" [^%nl]) / [^"%nl])*} '"' }) -> String
|
||||||
|
|
||||||
longstring <- ({ '".."' %ws?
|
longstring <- ({ '".."' %ws?
|
||||||
{|
|
{| (longstring_line (indent
|
||||||
(("|" {| ({("\\" / (!string_interpolation [^%nl]))+} / string_interpolation)* |})
|
longstring_line (nodent longstring_line)*
|
||||||
/ %line_comment)?
|
(dedent / longstring_error))?)
|
||||||
(%indent
|
/(indent
|
||||||
(%new_line "|" {|
|
longstring_line (nodent longstring_line)*
|
||||||
({("\\" / (!string_interpolation [^%nl]))+} / string_interpolation)*
|
(dedent / longstring_error)) |} }) -> Longstring
|
||||||
|})+
|
longstring_line <- "|" {| ({("\\" / (!string_interpolation [^%nl]))+} / string_interpolation)* |}
|
||||||
((%dedent (%new_line '..')?) / errors))?
|
longstring_error <- (({.+} ("" -> "Error while parsing Longstring")) => error)
|
||||||
|}}) -> Longstring
|
string_interpolation <- "\" %ws? (inline_functioncall / expression) %ws? "\"
|
||||||
string_interpolation <- "\" %ws? (functioncall / expression) %ws? "\"
|
|
||||||
number <- ({ {'-'? [0-9]+ ("." [0-9]+)?} }) -> Number
|
number <- ({ {"-"? (([0-9]+ "." [0-9]+) / ("." [0-9]+) / ([0-9]+)) } }) -> Number
|
||||||
|
|
||||||
|
-- Hack to allow %foo's to parse as "%foo" and "'s" separately
|
||||||
variable <- ({ ("%" {%wordchar (!"'" %wordchar)*}) }) -> Var
|
variable <- ({ ("%" {%wordchar (!"'" %wordchar)*}) }) -> Var
|
||||||
|
|
||||||
subexpression <-
|
|
||||||
("(" %ws? (functioncall / expression) %ws? ")")
|
|
||||||
/ ("(..)" %ws? %line_comment? %indent %new_line ((({ {| indented_fn_bits |} }) -> FunctionCall) / expression) %dedent (%new_line "..")?)
|
|
||||||
|
|
||||||
list <- ({ {|
|
list <- ({ {|
|
||||||
("[..]" %ws? %line_comment? %indent %new_line indented_list ","? ((%dedent (%new_line "..")?) / errors))
|
("[..]" indent
|
||||||
/ ("[" %ws? (list_items ","?)? %ws?"]")
|
list_line (nodent list_line)*
|
||||||
|
(dedent / (({.+} ("" -> "Error while parsing list")) => error)))
|
||||||
|
/("[" %ws? (list_line %ws?)? "]")
|
||||||
|} }) -> List
|
|} }) -> List
|
||||||
list_items <- ((functioncall / expression) (list_sep list_items)?)
|
list_line <- list_bit (%ws? "," tok_gap list_bit)* (%ws? ",")?
|
||||||
list_sep <- %ws? "," %ws?
|
list_bit <- inline_functioncall / expression
|
||||||
indented_list <-
|
|
||||||
(functioncall / expression) (((list_sep (%line_comment? %new_line)?) / (%line_comment? %new_line)) indented_list)?
|
block_comment <- "#.." [^%nl]* indent [^%nl]* (%nl ((%ws? (!. / &%nl)) / (!%dedented [^%nl]*)))*
|
||||||
|
line_comment <- "#" [^%nl]*
|
||||||
|
|
||||||
|
eol <- %ws? line_comment? (!. / &%nl)
|
||||||
|
ignored_line <- (%nodented (block_comment / line_comment)) / (%ws? (!. / &%nl))
|
||||||
|
indent <- eol (%nl ignored_line)* %nl %indented
|
||||||
|
nodent <- eol (%nl ignored_line)* %nl %nodented
|
||||||
|
dedent <- eol (%nl ignored_line)* (((!.) &%dedented) / (&(%nl %dedented)))
|
||||||
|
tok_gap <- %ws / %prev_edge / &("[" / [.,:;{("#%'])
|
||||||
|
dotdot <- nodent ".." %ws?
|
||||||
]=]
|
]=]
|
||||||
lingo = make_parser(lingo)
|
local whitespace = S(" \t") ^ 1
|
||||||
|
local defs = {
|
||||||
|
ws = whitespace,
|
||||||
|
nl = P("\n"),
|
||||||
|
wordchar = P(1) - S(' \t\n\r%#:;,.{}[]()"\\'),
|
||||||
|
indented = Cmt(S(" \t") ^ 0 * (#(P(1) - S(" \t\n") + (-P(1)))), check_indent),
|
||||||
|
nodented = Cmt(S(" \t") ^ 0 * (#(P(1) - S(" \t\n") + (-P(1)))), check_nodent),
|
||||||
|
dedented = Cmt(S(" \t") ^ 0 * (#(P(1) - S(" \t\n") + (-P(1)))), check_dedent),
|
||||||
|
prev_edge = B(S(" \t\n.,:;}])\"")),
|
||||||
|
error = function(src, pos, errors, err_msg)
|
||||||
|
local line_no = 1
|
||||||
|
for _ in src:sub(1, -#errors):gmatch("\n") do
|
||||||
|
line_no = line_no + 1
|
||||||
|
end
|
||||||
|
local err_pos = #src - #errors + 1
|
||||||
|
if errors:sub(1, 1) == "\n" then
|
||||||
|
err_pos = err_pos + #errors:match("[ \t]*", 2)
|
||||||
|
end
|
||||||
|
local start_of_err_line = err_pos
|
||||||
|
while src:sub(start_of_err_line, start_of_err_line) ~= "\n" and start_of_err_line > 1 do
|
||||||
|
start_of_err_line = start_of_err_line - 1
|
||||||
|
end
|
||||||
|
local start_of_prev_line = start_of_err_line - 1
|
||||||
|
while src:sub(start_of_prev_line, start_of_prev_line) ~= "\n" and start_of_prev_line > 1 do
|
||||||
|
start_of_prev_line = start_of_prev_line - 1
|
||||||
|
end
|
||||||
|
local prev_line, err_line, next_line
|
||||||
|
prev_line, err_line, next_line = src:match("([^\n]*)\n([^\n]*)\n([^\n]*)", start_of_prev_line + 1)
|
||||||
|
local pointer = ("-"):rep(err_pos - start_of_err_line + 0) .. "^"
|
||||||
|
return error("\n" .. tostring(err_msg or "Parse error") .. " in " .. tostring(filename) .. " on line " .. tostring(line_no) .. ":\n\n" .. tostring(prev_line) .. "\n" .. tostring(err_line) .. "\n" .. tostring(pointer) .. "\n" .. tostring(next_line) .. "\n")
|
||||||
|
end
|
||||||
|
}
|
||||||
|
setmetatable(defs, {
|
||||||
|
__index = function(t, key)
|
||||||
|
local fn
|
||||||
|
fn = function(src, value, errors)
|
||||||
|
local token = {
|
||||||
|
type = key,
|
||||||
|
src = src,
|
||||||
|
value = value,
|
||||||
|
errors = errors
|
||||||
|
}
|
||||||
|
return token
|
||||||
|
end
|
||||||
|
t[key] = fn
|
||||||
|
return fn
|
||||||
|
end
|
||||||
|
})
|
||||||
|
lingo = re.compile(lingo, defs)
|
||||||
local tree = lingo:match(str:gsub("\r", "") .. "\n")
|
local tree = lingo:match(str:gsub("\r", "") .. "\n")
|
||||||
if self.debug then
|
if self.debug then
|
||||||
self:writeln("\nPARSE TREE:")
|
self:writeln("\nPARSE TREE:")
|
||||||
@ -428,6 +418,9 @@ do
|
|||||||
kind = "Expression"
|
kind = "Expression"
|
||||||
end
|
end
|
||||||
assert(tree, "No tree provided.")
|
assert(tree, "No tree provided.")
|
||||||
|
if not tree.type then
|
||||||
|
self:error("Invalid tree: " .. tostring(utils.repr(tree)))
|
||||||
|
end
|
||||||
local indent = ""
|
local indent = ""
|
||||||
local buffer = { }
|
local buffer = { }
|
||||||
local return_value = nil
|
local return_value = nil
|
||||||
@ -485,8 +478,6 @@ do
|
|||||||
else
|
else
|
||||||
add("ret = " .. (to_lua(tree.value):match("%s*(.*)")))
|
add("ret = " .. (to_lua(tree.value):match("%s*(.*)")))
|
||||||
end
|
end
|
||||||
elseif "Expression" == _exp_0 then
|
|
||||||
add(to_lua(tree.value))
|
|
||||||
elseif "FunctionCall" == _exp_0 then
|
elseif "FunctionCall" == _exp_0 then
|
||||||
local name = self:fn_name_from_tree(tree)
|
local name = self:fn_name_from_tree(tree)
|
||||||
if self.defs[name] and self.defs[name].is_macro then
|
if self.defs[name] and self.defs[name].is_macro then
|
||||||
@ -670,8 +661,6 @@ do
|
|||||||
self:_yield_tree(tree.value, indent_level + 1)
|
self:_yield_tree(tree.value, indent_level + 1)
|
||||||
elseif "Statement" == _exp_0 then
|
elseif "Statement" == _exp_0 then
|
||||||
self:_yield_tree(tree.value, indent_level)
|
self:_yield_tree(tree.value, indent_level)
|
||||||
elseif "Expression" == _exp_0 then
|
|
||||||
self:_yield_tree(tree.value, indent_level)
|
|
||||||
elseif "FunctionCall" == _exp_0 then
|
elseif "FunctionCall" == _exp_0 then
|
||||||
local name = self:fn_name_from_tree(tree)
|
local name = self:fn_name_from_tree(tree)
|
||||||
local args
|
local args
|
||||||
@ -737,14 +726,14 @@ do
|
|||||||
end
|
end
|
||||||
return table.concat(result, "\n")
|
return table.concat(result, "\n")
|
||||||
end,
|
end,
|
||||||
compile = function(self, src, output_file)
|
compile = function(self, src, filename, output_file)
|
||||||
if output_file == nil then
|
if output_file == nil then
|
||||||
output_file = nil
|
output_file = nil
|
||||||
end
|
end
|
||||||
if self.debug then
|
if self.debug then
|
||||||
self:writeln("COMPILING:\n" .. tostring(src))
|
self:writeln("COMPILING:\n" .. tostring(src))
|
||||||
end
|
end
|
||||||
local tree = self:parse(src)
|
local tree = self:parse(src, filename)
|
||||||
assert(tree, "Tree failed to compile: " .. tostring(src))
|
assert(tree, "Tree failed to compile: " .. tostring(src))
|
||||||
local code, retval = self:tree_to_lua(tree)
|
local code, retval = self:tree_to_lua(tree)
|
||||||
if output_file then
|
if output_file then
|
||||||
@ -764,7 +753,7 @@ do
|
|||||||
self.callstack = { }
|
self.callstack = { }
|
||||||
return error()
|
return error()
|
||||||
end,
|
end,
|
||||||
test = function(self, src, expected)
|
test = function(self, src, filename, expected)
|
||||||
local i = 1
|
local i = 1
|
||||||
while i ~= nil do
|
while i ~= nil do
|
||||||
local start, stop = src:find("\n\n", i)
|
local start, stop = src:find("\n\n", i)
|
||||||
@ -777,7 +766,7 @@ do
|
|||||||
local test_src
|
local test_src
|
||||||
test_src, expected = test:sub(1, start - 1), test:sub(stop + 1, -1)
|
test_src, expected = test:sub(1, start - 1), test:sub(stop + 1, -1)
|
||||||
expected = expected:match('[\n]*(.*[^\n])')
|
expected = expected:match('[\n]*(.*[^\n])')
|
||||||
local tree = self:parse(test_src)
|
local tree = self:parse(test_src, filename)
|
||||||
local got = self:stringify_tree(tree.value.body)
|
local got = self:stringify_tree(tree.value.body)
|
||||||
if got ~= expected then
|
if got ~= expected then
|
||||||
self:error("TEST FAILED!\nSource:\n" .. tostring(test_src) .. "\nExpected:\n" .. tostring(expected) .. "\n\nGot:\n" .. tostring(got))
|
self:error("TEST FAILED!\nSource:\n" .. tostring(test_src) .. "\nExpected:\n" .. tostring(expected) .. "\n\nGot:\n" .. tostring(got))
|
||||||
@ -819,28 +808,34 @@ do
|
|||||||
self:def("require %filename", function(self, vars)
|
self:def("require %filename", function(self, vars)
|
||||||
if not self.loaded_files[vars.filename] then
|
if not self.loaded_files[vars.filename] then
|
||||||
local file = io.open(vars.filename)
|
local file = io.open(vars.filename)
|
||||||
self.loaded_files[vars.filename] = self:run(file:read('*a'))
|
if not file then
|
||||||
|
self:error("File does not exist: " .. tostring(vars.filename))
|
||||||
|
end
|
||||||
|
self.loaded_files[vars.filename] = self:run(file:read('*a'), vars.filename)
|
||||||
end
|
end
|
||||||
return self.loaded_files[vars.filename]
|
return self.loaded_files[vars.filename]
|
||||||
end)
|
end)
|
||||||
return self:def("run file %filename", function(self, vars)
|
return self:def("run file %filename", function(self, vars)
|
||||||
local file = io.open(vars.filename)
|
local file = io.open(vars.filename)
|
||||||
return self:run(file:read('*a'))
|
if not file then
|
||||||
|
self:error("File does not exist: " .. tostring(vars.filename))
|
||||||
|
end
|
||||||
|
return self:run(file:read('*a'), vars.filename)
|
||||||
end)
|
end)
|
||||||
end
|
end
|
||||||
}
|
}
|
||||||
_base_0.__index = _base_0
|
_base_0.__index = _base_0
|
||||||
_class_0 = setmetatable({
|
_class_0 = setmetatable({
|
||||||
__init = function(self, parent)
|
__init = function(self, parent)
|
||||||
|
self.write = function(self, ...)
|
||||||
|
return io.write(...)
|
||||||
|
end
|
||||||
self.defs = setmetatable({ }, {
|
self.defs = setmetatable({ }, {
|
||||||
__index = parent and parent.defs
|
__index = parent and parent.defs
|
||||||
})
|
})
|
||||||
self.callstack = { }
|
self.callstack = { }
|
||||||
self.debug = false
|
self.debug = false
|
||||||
self:initialize_core()
|
self:initialize_core()
|
||||||
self.write = function(self, ...)
|
|
||||||
return io.write(...)
|
|
||||||
end
|
|
||||||
self.utils = utils
|
self.utils = utils
|
||||||
self.loaded_files = { }
|
self.loaded_files = { }
|
||||||
end,
|
end,
|
||||||
@ -886,7 +881,7 @@ if arg and arg[1] then
|
|||||||
if arg[2] == "-" then
|
if arg[2] == "-" then
|
||||||
c.write = function() end
|
c.write = function() end
|
||||||
end
|
end
|
||||||
local code, retval = c:compile(input)
|
local code, retval = c:compile(input, arg[1])
|
||||||
c.write = _write
|
c.write = _write
|
||||||
if arg[2] then
|
if arg[2] then
|
||||||
local output
|
local output
|
||||||
@ -907,7 +902,7 @@ if arg and arg[1] then
|
|||||||
end
|
end
|
||||||
elseif arg then
|
elseif arg then
|
||||||
local c = NomsuCompiler()
|
local c = NomsuCompiler()
|
||||||
c:run('run file "core.nom"')
|
c:run('require "lib/core.nom"')
|
||||||
while true do
|
while true do
|
||||||
local buff = ""
|
local buff = ""
|
||||||
while true do
|
while true do
|
||||||
|
14
utils.moon
14
utils.moon
@ -51,14 +51,28 @@ utils = {
|
|||||||
start,stop,step = 1,start,1
|
start,stop,step = 1,start,1
|
||||||
elseif step == nil
|
elseif step == nil
|
||||||
step = 1
|
step = 1
|
||||||
|
elseif step == 0
|
||||||
|
error("Range step cannot be zero.")
|
||||||
return setmetatable({:start,:stop,:step}, {
|
return setmetatable({:start,:stop,:step}, {
|
||||||
__ipairs: =>
|
__ipairs: =>
|
||||||
iter = (i)=>
|
iter = (i)=>
|
||||||
if i <= (@stop-@start)/@step
|
if i <= (@stop-@start)/@step
|
||||||
return i+1, @start+i*@step
|
return i+1, @start+i*@step
|
||||||
return iter, @, 0
|
return iter, @, 0
|
||||||
|
__index: (i)=>
|
||||||
|
if type(i) != "Number" then return nil
|
||||||
|
if i % 1 != 0 then return nil
|
||||||
|
if i <= 0 or i-1 > (@stop-@start)/@step then return nil
|
||||||
|
return @start + (i-1)*@step
|
||||||
|
__len: =>
|
||||||
|
len = (@stop-@start)/@step
|
||||||
|
if len < 0 then len = 0
|
||||||
|
return len
|
||||||
|
|
||||||
})
|
})
|
||||||
|
|
||||||
|
nth_to_last: (list, n) -> list[#list-n+1]
|
||||||
|
|
||||||
keys: (t)-> [k for k in pairs(t)]
|
keys: (t)-> [k for k in pairs(t)]
|
||||||
values: (t)-> [v for _,v in pairs(t)]
|
values: (t)-> [v for _,v in pairs(t)]
|
||||||
set: (list)-> {i,true for i in *list}
|
set: (list)-> {i,true for i in *list}
|
||||||
|
Loading…
Reference in New Issue
Block a user