Some stuff changed to allow escaped args and some other ports from the
two_defs branch.
This commit is contained in:
parent
8c0816995a
commit
b3b8c4d731
@ -93,7 +93,7 @@ rule [dict from entries %items] =:
|
|||||||
%dict -> (%pair -> 1) = (%pair -> 2)
|
%dict -> (%pair -> 1) = (%pair -> 2)
|
||||||
%dict
|
%dict
|
||||||
|
|
||||||
compile [dict %items] to:
|
compile [dict %items, d %items] to:
|
||||||
if ((%items's "type") == "Thunk"):
|
if ((%items's "type") == "Thunk"):
|
||||||
%item_codes = []
|
%item_codes = []
|
||||||
for %func_call in (%items's "value"):
|
for %func_call in (%items's "value"):
|
||||||
|
@ -218,11 +218,48 @@ compile [when %branch_value == ? %body] to code:
|
|||||||
|end --when == ?
|
|end --when == ?
|
||||||
%result
|
%result
|
||||||
|
|
||||||
# With statement
|
# Try/except
|
||||||
compile [with %thing = %value %action] to code: ".."
|
compile [..]
|
||||||
|
try %action and if it succeeds %success or if it fails %fallback
|
||||||
|
try %action and if it fails %fallback or if it succeeds %success
|
||||||
|
..to code: ".."
|
||||||
|do
|
|do
|
||||||
| local old_value = \(%thing as lua);
|
| local fell_through = false;
|
||||||
| \(%thing as lua) = \(%value as lua);
|
| local ok, ret1, ret2 = pcall(function(nomsu, vars)
|
||||||
| \(%action as lua statements);
|
| \(%action as lua statements)
|
||||||
| \(%thing as lua) = old_value;
|
| fell_through = true;
|
||||||
|
| end, nomsu, vars);
|
||||||
|
| if ok then
|
||||||
|
| \(%success as lua statements)
|
||||||
|
| end
|
||||||
|
| if not ok then
|
||||||
|
| \(%fallback as lua statements)
|
||||||
|
| elseif not fell_through then
|
||||||
|
| return ret1, ret2;
|
||||||
|
| end
|
||||||
|end
|
|end
|
||||||
|
parse [try %action] as:
|
||||||
|
try %action and if it succeeds {pass} or if it fails {pass}
|
||||||
|
parse [try %action and if it fails %fallback] as:
|
||||||
|
try %action and if it succeeds {pass} or if it fails %fallback
|
||||||
|
parse [try %action and if it succeeds %success] as:
|
||||||
|
try %action and if it succeeds %success or if it fails {pass}
|
||||||
|
|
||||||
|
# Do/finally:
|
||||||
|
compile [do %action then always %final_action] to code: ".."
|
||||||
|
|do
|
||||||
|
| local fell_through = false;
|
||||||
|
| local ok, ret1, ret2 = pcall(function(nomsu, vars)
|
||||||
|
| \(%action as lua statements)
|
||||||
|
| fell_through = true;
|
||||||
|
| end, nomsu, vars);
|
||||||
|
| local ok2, _ = pcall(function(nomsu, vars)
|
||||||
|
| \(%final_action as lua statements)
|
||||||
|
| end, nomsu, vars);
|
||||||
|
| if not ok then nomsu:error(ret1); end
|
||||||
|
| if not ok2 then nomsu:error(ret2); end
|
||||||
|
| if not fell_through then
|
||||||
|
| return ret1, ret2;
|
||||||
|
| end
|
||||||
|
|end
|
||||||
|
|
||||||
|
@ -3,52 +3,16 @@ require "lib/control_flow.nom"
|
|||||||
require "lib/operators.nom"
|
require "lib/operators.nom"
|
||||||
require "lib/collections.nom"
|
require "lib/collections.nom"
|
||||||
|
|
||||||
# Permission functions
|
rule [called by %whitelist] =:
|
||||||
rule [standardize rules %rules] =:
|
if ((%whitelist's "type") != "List"): %whitelist = [%whitelist]
|
||||||
if ((type of %rules) == "string"): %rules = [%rules]
|
%defs = (..)
|
||||||
%stubs = (nomsu "get_stubs" [%rules])
|
dict ([(nomsu's "defs")->(nomsu "get_stub" [%]), yes] for all %whitelist)
|
||||||
%result = []
|
for %caller in (nomsu's "callstack"):
|
||||||
for %stub in %stubs:
|
if (%caller == "#macro"): do next %caller
|
||||||
%def = ((nomsu's "defs")->%stub)
|
if (%defs -> (nomsu "get_stub" [%caller's 1])): return (yes)
|
||||||
if %def:
|
return (no)
|
||||||
%aliases = (%def's "aliases")
|
|
||||||
for all %aliases: add % to %result
|
|
||||||
..else: add %def to %result
|
|
||||||
unique %result
|
|
||||||
|
|
||||||
rule [restrict %rules to within %elite_rules] =:
|
parse [fail unless called by %whitelist] as:
|
||||||
%rules = (standardize rules %rules)
|
unless (called by %whitelist): error "Failed to find \(%whitelist) in callstack."
|
||||||
%elite_rules = (standardize rules %elite_rules)
|
|
||||||
for all (flatten [%elite_rules, %rules]):
|
|
||||||
assert ((nomsu's "defs") has key %) "Undefined function: \(%)"
|
|
||||||
for %rule in %rules:
|
|
||||||
assert (nomsu "check_permission" [%]) ".."
|
|
||||||
|You do not have permission to restrict permissions for function: \(%)
|
|
||||||
((nomsu) ->* ["defs",%rule,"whiteset"]) = (..)
|
|
||||||
dict ([%, yes] for all %elite_rules)
|
|
||||||
|
|
||||||
rule [allow %elite_rules to use %rules] =:
|
|
||||||
%rules = (standardize rules %rules)
|
|
||||||
%elite_rules = (standardize rules %elite_rules)
|
|
||||||
for all (flatten [%elite_rules, %rules]):
|
|
||||||
assert ((nomsu's "defs") has key %) "Undefined function: \(%)"
|
|
||||||
for %rule in %rules:
|
|
||||||
assert (nomsu "check_permission" [%rule]) ".."
|
|
||||||
|You do not have permission to grant permissions for function: \(%rule)
|
|
||||||
%whiteset = ((nomsu) ->* ["defs",%rule,"whiteset"])
|
|
||||||
if (not %whiteset): go to next %rule
|
|
||||||
for all %elite_rules: %whiteset -> % = (yes)
|
|
||||||
|
|
||||||
rule [forbid %pleb_rules to use %rules] =:
|
|
||||||
%rules = (standardize rules %rules)
|
|
||||||
%pleb_rules = (standardize rules %pleb_rules)
|
|
||||||
for all (flatten [%pleb_rules, %used]):
|
|
||||||
assert ((nomsu's "defs") has key %) "Undefined function: \(%)"
|
|
||||||
for all %rules:
|
|
||||||
assert (nomsu "check_permission" [%]) ".."
|
|
||||||
|You do not have permission to grant permissions for function: \(%)
|
|
||||||
%whiteset = ((nomsu) ->* ["defs",%,"whiteset"])
|
|
||||||
assert %whiteset ".."
|
|
||||||
|Cannot individually restrict permissions for \(%) because it is currently
|
|
||||||
|available to everyone. Perhaps you meant to use "restrict % to within %" instead?
|
|
||||||
for all %pleb_rules: %whiteset's % = (nil)
|
|
||||||
|
@ -130,24 +130,3 @@ lua> ".."
|
|||||||
| end;
|
| end;
|
||||||
|end;
|
|end;
|
||||||
|
|
||||||
compile [try %action and if it fails %fallback] to code: ".."
|
|
||||||
|do
|
|
||||||
| local _write_err = nomsu.write_err
|
|
||||||
| nomsu.write_err = function() end;
|
|
||||||
| local had_return = true;
|
|
||||||
| local ok, ret1, ret2 = pcall(function(nomsu, vars)
|
|
||||||
| local ret
|
|
||||||
| \(%action as lua statements)
|
|
||||||
| had_return = false;
|
|
||||||
| return ret;
|
|
||||||
| end, nomsu, vars);
|
|
||||||
| nomsu.write_err = _write_err;
|
|
||||||
| if not ok then
|
|
||||||
| \(%fallback as lua statements)
|
|
||||||
| elseif had_return then
|
|
||||||
| return ret1, ret2;
|
|
||||||
| else
|
|
||||||
| ret = ret1;
|
|
||||||
| end
|
|
||||||
|end
|
|
||||||
|
|
||||||
|
@ -2,9 +2,51 @@ require "lib/metaprogramming.nom"
|
|||||||
require "lib/utils.nom"
|
require "lib/utils.nom"
|
||||||
require "lib/control_flow.nom"
|
require "lib/control_flow.nom"
|
||||||
require "lib/operators.nom"
|
require "lib/operators.nom"
|
||||||
|
require "lib/collections.nom"
|
||||||
|
|
||||||
|
|
||||||
compile [say %str] to:
|
compile [say %str] to:
|
||||||
if ((%str's "type") == "String"):
|
if ((%str's "type") == "String"):
|
||||||
"nomsu:writeln(\(%str as lua))"
|
"nomsu:writeln(\(%str as lua))"
|
||||||
..else:
|
..else:
|
||||||
"nomsu:writeln(nomsu:stringify(\(%str as lua)))"
|
"nomsu:writeln(nomsu:stringify(\(%str as lua)))"
|
||||||
|
|
||||||
|
compile [do %action] to code:
|
||||||
|
if ((%action's "type") == "Thunk"):
|
||||||
|
%action as lua statements
|
||||||
|
..else:
|
||||||
|
"(\(%action as lua))(nomsu, vars);"
|
||||||
|
|
||||||
|
# With statement
|
||||||
|
compile [with %assignments %action] to code:
|
||||||
|
%data = []
|
||||||
|
for %i = %assignment in (%assignments' "value"):
|
||||||
|
%tokens = (%assignment's "value")
|
||||||
|
%var = (%tokens -> 1)
|
||||||
|
%eq = (%tokens -> 2)
|
||||||
|
assert (=lua "vars.eq and vars.eq.type == 'Word' and vars.eq.value == '='") ".."
|
||||||
|
|Invalid format for 'with' statement. List entries must have the form %var = (value)
|
||||||
|
%value = (%tokens -> 3)
|
||||||
|
add (d{i=%i; var=%var; value=%value}) to %data
|
||||||
|
%foo = (..)
|
||||||
|
join (..)
|
||||||
|
"local old_value\(%->"i") = \((%->"var") as lua); \((%->"var") as lua) = \((%->"value") as lua);"
|
||||||
|
..for all %data
|
||||||
|
..with glue "\n "
|
||||||
|
".."
|
||||||
|
|do
|
||||||
|
| \(%foo)
|
||||||
|
| local fell_through = false;
|
||||||
|
| local ok, ret1, ret2 = pcall(function(nomsu, vars)
|
||||||
|
| \(%action as lua statements);
|
||||||
|
| fell_through = true;
|
||||||
|
| end, nomsu, vars);
|
||||||
|
| \(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
|
||||||
|
|
||||||
|
|
||||||
|
376
nomsu.lua
376
nomsu.lua
@ -19,6 +19,19 @@ do
|
|||||||
local _obj_0 = table
|
local _obj_0 = table
|
||||||
insert, remove, concat = _obj_0.insert, _obj_0.remove, _obj_0.concat
|
insert, remove, concat = _obj_0.insert, _obj_0.remove, _obj_0.concat
|
||||||
end
|
end
|
||||||
|
if _VERSION == "Lua 5.1" then
|
||||||
|
local xp = xpcall
|
||||||
|
local xpcall
|
||||||
|
xpcall = function(f, errhandler, ...)
|
||||||
|
local args = {
|
||||||
|
n = select("#", ...),
|
||||||
|
...
|
||||||
|
}
|
||||||
|
return xp(function(...)
|
||||||
|
return f(unpack(args, 1, args.n))
|
||||||
|
end), errhandler
|
||||||
|
end
|
||||||
|
end
|
||||||
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
|
||||||
@ -166,12 +179,23 @@ local defs = {
|
|||||||
return pos, tostring(CURRENT_FILE) .. ":" .. tostring(line_no)
|
return pos, tostring(CURRENT_FILE) .. ":" .. tostring(line_no)
|
||||||
end,
|
end,
|
||||||
FunctionCall = function(src, line_no, value, errors)
|
FunctionCall = function(src, line_no, value, errors)
|
||||||
|
local stub = concat((function()
|
||||||
|
local _accum_0 = { }
|
||||||
|
local _len_0 = 1
|
||||||
|
for _index_0 = 1, #value do
|
||||||
|
local t = value[_index_0]
|
||||||
|
_accum_0[_len_0] = (t.type == "Word" and t.value or "%")
|
||||||
|
_len_0 = _len_0 + 1
|
||||||
|
end
|
||||||
|
return _accum_0
|
||||||
|
end)(), " ")
|
||||||
return {
|
return {
|
||||||
type = "FunctionCall",
|
type = "FunctionCall",
|
||||||
src = src,
|
src = src,
|
||||||
line_no = line_no,
|
line_no = line_no,
|
||||||
value = value,
|
value = value,
|
||||||
errors = errors
|
errors = errors,
|
||||||
|
stub = stub
|
||||||
}
|
}
|
||||||
end,
|
end,
|
||||||
error = function(src, pos, errors, err_msg)
|
error = function(src, pos, errors, err_msg)
|
||||||
@ -241,12 +265,22 @@ do
|
|||||||
end
|
end
|
||||||
assert(type(thunk) == 'function', "Bad thunk: " .. tostring(repr(thunk)))
|
assert(type(thunk) == 'function', "Bad thunk: " .. tostring(repr(thunk)))
|
||||||
local canonical_args = nil
|
local canonical_args = nil
|
||||||
|
local canonical_escaped_args = nil
|
||||||
local aliases = { }
|
local aliases = { }
|
||||||
self.def_number = self.def_number + 1
|
self.__class.def_number = self.__class.def_number + 1
|
||||||
|
local def = {
|
||||||
|
thunk = thunk,
|
||||||
|
src = src,
|
||||||
|
is_macro = is_macro,
|
||||||
|
aliases = { },
|
||||||
|
def_number = self.__class.def_number,
|
||||||
|
defs = self.defs
|
||||||
|
}
|
||||||
|
local where_defs_go = ((getmetatable(self.defs) or { }).__newindex) or self.defs
|
||||||
for _index_0 = 1, #signature do
|
for _index_0 = 1, #signature do
|
||||||
local _des_0 = signature[_index_0]
|
local _des_0 = signature[_index_0]
|
||||||
local stub, arg_names
|
local stub, arg_names, escaped_args
|
||||||
stub, arg_names = _des_0[1], _des_0[2]
|
stub, arg_names, escaped_args = _des_0[1], _des_0[2], _des_0[3]
|
||||||
assert(stub, "NO STUB FOUND: " .. tostring(repr(signature)))
|
assert(stub, "NO STUB FOUND: " .. tostring(repr(signature)))
|
||||||
if self.debug then
|
if self.debug then
|
||||||
self:writeln(tostring(colored.bright("DEFINING RULE:")) .. " " .. tostring(colored.underscore(colored.magenta(repr(stub)))) .. " " .. tostring(colored.bright("WITH ARGS")) .. " " .. tostring(colored.dim(repr(arg_names))))
|
self:writeln(tostring(colored.bright("DEFINING RULE:")) .. " " .. tostring(colored.underscore(colored.magenta(repr(stub)))) .. " " .. tostring(colored.bright("WITH ARGS")) .. " " .. tostring(colored.dim(repr(arg_names))))
|
||||||
@ -263,38 +297,104 @@ do
|
|||||||
else
|
else
|
||||||
canonical_args = utils.set(arg_names)
|
canonical_args = utils.set(arg_names)
|
||||||
end
|
end
|
||||||
insert(aliases, stub)
|
if canonical_escaped_args then
|
||||||
self.defs[stub] = {
|
assert(utils.equivalent(escaped_args, canonical_escaped_args), "Mismatched escaped args")
|
||||||
thunk = thunk,
|
else
|
||||||
|
canonical_escaped_args = escaped_args
|
||||||
|
def.escaped_args = escaped_args
|
||||||
|
end
|
||||||
|
insert(def.aliases, stub)
|
||||||
|
local stub_def = setmetatable({
|
||||||
stub = stub,
|
stub = stub,
|
||||||
arg_names = arg_names,
|
arg_names = arg_names,
|
||||||
src = src,
|
escaped_args = escaped_args
|
||||||
is_macro = is_macro,
|
}, {
|
||||||
aliases = aliases,
|
__index = def
|
||||||
def_number = self.def_number
|
})
|
||||||
}
|
rawset(where_defs_go, stub, stub_def)
|
||||||
end
|
end
|
||||||
end,
|
end,
|
||||||
defmacro = function(self, signature, thunk, src)
|
defmacro = function(self, signature, thunk, src)
|
||||||
return self:def(signature, thunk, src, true)
|
return self:def(signature, thunk, src, true)
|
||||||
end,
|
end,
|
||||||
serialize_defs = function(self, after)
|
scoped = function(self, thunk)
|
||||||
|
local old_defs = self.defs
|
||||||
|
self.defs = setmetatable({ }, {
|
||||||
|
__index = old_defs
|
||||||
|
})
|
||||||
|
local ok, ret1, ret2 = pcall(thunk, self)
|
||||||
|
self.defs = old_defs
|
||||||
|
if not ok then
|
||||||
|
self:error(ret1)
|
||||||
|
end
|
||||||
|
return ret1, ret2
|
||||||
|
end,
|
||||||
|
serialize_defs = function(self, scope, after)
|
||||||
|
if scope == nil then
|
||||||
|
scope = nil
|
||||||
|
end
|
||||||
if after == nil then
|
if after == nil then
|
||||||
after = 0
|
after = 0
|
||||||
end
|
end
|
||||||
defs = { }
|
scope = scope or self.defs
|
||||||
for _, def in pairs(self.defs) do
|
local defs_by_num = { }
|
||||||
defs[def.def_number] = def.src or ""
|
for stub, def in pairs(scope) do
|
||||||
|
if def and stub:sub(1, 1) ~= "#" then
|
||||||
|
defs_by_num[def.def_number] = def
|
||||||
|
end
|
||||||
|
end
|
||||||
|
local keys
|
||||||
|
do
|
||||||
|
local _accum_0 = { }
|
||||||
|
local _len_0 = 1
|
||||||
|
for k, v in pairs(defs_by_num) do
|
||||||
|
_accum_0[_len_0] = k
|
||||||
|
_len_0 = _len_0 + 1
|
||||||
|
end
|
||||||
|
keys = _accum_0
|
||||||
end
|
end
|
||||||
local keys = utils.keys(defs)
|
|
||||||
table.sort(keys)
|
table.sort(keys)
|
||||||
local buff = { }
|
local buff = { }
|
||||||
for _index_0 = 1, #keys do
|
local k_i = 1
|
||||||
local i = keys[_index_0]
|
local _using = nil
|
||||||
if i > after and #defs[i] > 0 then
|
local _using_do = { }
|
||||||
insert(buff, defs[i])
|
for k_i, i in ipairs(keys) do
|
||||||
|
local _continue_0 = false
|
||||||
|
repeat
|
||||||
|
if i <= after then
|
||||||
|
_continue_0 = true
|
||||||
|
break
|
||||||
|
end
|
||||||
|
local def = defs_by_num[i]
|
||||||
|
if def.defs == scope then
|
||||||
|
if def.src then
|
||||||
|
insert(buff, def.src)
|
||||||
|
end
|
||||||
|
_continue_0 = true
|
||||||
|
break
|
||||||
|
end
|
||||||
|
if _using == def.defs then
|
||||||
|
if def.src then
|
||||||
|
insert(_using_do, def.src)
|
||||||
|
end
|
||||||
|
else
|
||||||
|
_using = def.defs
|
||||||
|
_using_do = {
|
||||||
|
def.src
|
||||||
|
}
|
||||||
|
end
|
||||||
|
if k_i == #keys or defs_by_num[keys[k_i + 1]].defs ~= _using then
|
||||||
|
insert(buff, "using:\n " .. tostring(self:indent(self:serialize_defs(_using))) .. "\n..do:\n " .. tostring(self:indent(concat(_using_do, "\n"))))
|
||||||
|
end
|
||||||
|
_continue_0 = true
|
||||||
|
until true
|
||||||
|
if not _continue_0 then
|
||||||
|
break
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
for k, v in pairs(scope["#vars"] or { }) do
|
||||||
|
insert(buff, "<@" .. tostring(k) .. "> = " .. tostring(self:value_to_nomsu(v)))
|
||||||
|
end
|
||||||
return concat(buff, "\n")
|
return concat(buff, "\n")
|
||||||
end,
|
end,
|
||||||
call = function(self, stub, line_no, ...)
|
call = function(self, stub, line_no, ...)
|
||||||
@ -306,7 +406,7 @@ do
|
|||||||
stub,
|
stub,
|
||||||
line_no
|
line_no
|
||||||
})
|
})
|
||||||
if def == nil then
|
if not (def) then
|
||||||
self:error("Attempt to call undefined function: " .. tostring(stub))
|
self:error("Attempt to call undefined function: " .. tostring(stub))
|
||||||
end
|
end
|
||||||
if not (def.is_macro) then
|
if not (def.is_macro) then
|
||||||
@ -326,14 +426,16 @@ do
|
|||||||
self:write(tostring(colored.bright("CALLING")) .. " " .. tostring(colored.magenta(colored.underscore(stub))) .. " ")
|
self:write(tostring(colored.bright("CALLING")) .. " " .. tostring(colored.magenta(colored.underscore(stub))) .. " ")
|
||||||
self:writeln(tostring(colored.bright("WITH ARGS:")) .. " " .. tostring(colored.dim(repr(args))))
|
self:writeln(tostring(colored.bright("WITH ARGS:")) .. " " .. tostring(colored.dim(repr(args))))
|
||||||
end
|
end
|
||||||
|
local old_defs
|
||||||
|
old_defs, self.defs = self.defs, def.defs
|
||||||
local rets = {
|
local rets = {
|
||||||
thunk(self, args)
|
thunk(self, args)
|
||||||
}
|
}
|
||||||
|
self.defs = old_defs
|
||||||
remove(self.callstack)
|
remove(self.callstack)
|
||||||
return unpack(rets)
|
return unpack(rets)
|
||||||
end,
|
end,
|
||||||
run_macro = function(self, tree)
|
run_macro = function(self, tree)
|
||||||
local stub = self:get_stub(tree)
|
|
||||||
local args
|
local args
|
||||||
do
|
do
|
||||||
local _accum_0 = { }
|
local _accum_0 = { }
|
||||||
@ -349,14 +451,55 @@ do
|
|||||||
args = _accum_0
|
args = _accum_0
|
||||||
end
|
end
|
||||||
if self.debug then
|
if self.debug then
|
||||||
self:write(tostring(colored.bright("RUNNING MACRO")) .. " " .. tostring(colored.underscore(colored.magenta(stub))) .. " ")
|
self:write(tostring(colored.bright("RUNNING MACRO")) .. " " .. tostring(colored.underscore(colored.magenta(tree.stub))) .. " ")
|
||||||
self:writeln(tostring(colored.bright("WITH ARGS:")) .. " " .. tostring(colored.dim(repr(args))))
|
self:writeln(tostring(colored.bright("WITH ARGS:")) .. " " .. tostring(colored.dim(repr(args))))
|
||||||
end
|
end
|
||||||
insert(self.callstack, "#macro")
|
insert(self.callstack, "#macro")
|
||||||
local expr, statement = self:call(stub, tree.line_no, unpack(args))
|
local expr, statement = self:call(tree.stub, tree.line_no, unpack(args))
|
||||||
remove(self.callstack)
|
remove(self.callstack)
|
||||||
return expr, statement
|
return expr, statement
|
||||||
end,
|
end,
|
||||||
|
dedent = function(self, code)
|
||||||
|
if not (code:find("\n")) then
|
||||||
|
return code
|
||||||
|
end
|
||||||
|
local spaces, indent_spaces = math.huge, math.huge
|
||||||
|
for line in code:gmatch("\n([^\n]*)") do
|
||||||
|
local _continue_0 = false
|
||||||
|
repeat
|
||||||
|
if line:match("^%s*#.*") then
|
||||||
|
_continue_0 = true
|
||||||
|
break
|
||||||
|
else
|
||||||
|
do
|
||||||
|
local s = line:match("^(%s*)%.%..*")
|
||||||
|
if s then
|
||||||
|
spaces = math.min(spaces, #s)
|
||||||
|
else
|
||||||
|
do
|
||||||
|
s = line:match("^(%s*)%S.*")
|
||||||
|
if s then
|
||||||
|
indent_spaces = math.min(indent_spaces, #s)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
_continue_0 = true
|
||||||
|
until true
|
||||||
|
if not _continue_0 then
|
||||||
|
break
|
||||||
|
end
|
||||||
|
end
|
||||||
|
if spaces ~= math.huge and spaces < indent_spaces then
|
||||||
|
return (code:gsub("\n" .. (" "):rep(spaces), "\n"))
|
||||||
|
else
|
||||||
|
return (code:gsub("\n" .. (" "):rep(indent_spaces), "\n "))
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
indent = function(self, code)
|
||||||
|
return (code:gsub("\n", "\n "))
|
||||||
|
end,
|
||||||
assert_permission = function(self, stub)
|
assert_permission = function(self, stub)
|
||||||
local fn_def = self.defs[stub]
|
local fn_def = self.defs[stub]
|
||||||
if not (fn_def) then
|
if not (fn_def) then
|
||||||
@ -451,7 +594,7 @@ do
|
|||||||
local ok, expr, statements = pcall(self.tree_to_lua, self, statement)
|
local ok, expr, statements = pcall(self.tree_to_lua, self, statement)
|
||||||
if not ok then
|
if not ok then
|
||||||
self:errorln(tostring(colored.red("Error occurred in statement:")) .. "\n" .. tostring(colored.bright(colored.yellow(statement.src))))
|
self:errorln(tostring(colored.red("Error occurred in statement:")) .. "\n" .. tostring(colored.bright(colored.yellow(statement.src))))
|
||||||
self:error(expr)
|
error(expr)
|
||||||
end
|
end
|
||||||
local code_for_statement = ([[return (function(nomsu, vars)
|
local code_for_statement = ([[return (function(nomsu, vars)
|
||||||
%s
|
%s
|
||||||
@ -480,7 +623,7 @@ end);]]):format(statements or "", expr or "ret")
|
|||||||
if not ok then
|
if not ok then
|
||||||
self:errorln(tostring(colored.red("Error occurred in statement:")) .. "\n" .. tostring(colored.yellow(statement.src)))
|
self:errorln(tostring(colored.red("Error occurred in statement:")) .. "\n" .. tostring(colored.yellow(statement.src)))
|
||||||
self:errorln(debug.traceback())
|
self:errorln(debug.traceback())
|
||||||
self:error(ret)
|
error(ret)
|
||||||
end
|
end
|
||||||
if statements then
|
if statements then
|
||||||
insert(buffer, statements)
|
insert(buffer, statements)
|
||||||
@ -514,10 +657,6 @@ end);]]):format(concat(buffer, "\n"))
|
|||||||
if force_inline == nil then
|
if force_inline == nil then
|
||||||
force_inline = false
|
force_inline = false
|
||||||
end
|
end
|
||||||
local indent
|
|
||||||
indent = function(s)
|
|
||||||
return s:gsub("\n", "\n ")
|
|
||||||
end
|
|
||||||
assert(tree, "No tree provided.")
|
assert(tree, "No tree provided.")
|
||||||
if not tree.type then
|
if not tree.type then
|
||||||
self:errorln(debug.traceback())
|
self:errorln(debug.traceback())
|
||||||
@ -553,7 +692,7 @@ end);]]):format(concat(buffer, "\n"))
|
|||||||
return _accum_0
|
return _accum_0
|
||||||
end)(), "; ")), true
|
end)(), "; ")), true
|
||||||
else
|
else
|
||||||
return ":" .. indent("\n" .. concat((function()
|
return ":" .. self:indent("\n" .. concat((function()
|
||||||
local _accum_0 = { }
|
local _accum_0 = { }
|
||||||
local _len_0 = 1
|
local _len_0 = 1
|
||||||
local _list_0 = tree.value
|
local _list_0 = tree.value
|
||||||
@ -569,17 +708,23 @@ end);]]):format(concat(buffer, "\n"))
|
|||||||
local buff = ""
|
local buff = ""
|
||||||
local sep = ""
|
local sep = ""
|
||||||
local inline = true
|
local inline = true
|
||||||
local do_arg
|
local line_len = 0
|
||||||
do_arg = function(arg) end
|
|
||||||
local _list_0 = tree.value
|
local _list_0 = tree.value
|
||||||
for _index_0 = 1, #_list_0 do
|
for _index_0 = 1, #_list_0 do
|
||||||
local arg = _list_0[_index_0]
|
local arg = _list_0[_index_0]
|
||||||
local arg_inline
|
local arg_inline
|
||||||
nomsu, arg_inline = self:tree_to_nomsu(arg, force_inline)
|
nomsu, arg_inline = self:tree_to_nomsu(arg, force_inline)
|
||||||
buff = buff .. sep
|
if sep == " " and line_len + #nomsu > 80 then
|
||||||
|
sep = "\n.."
|
||||||
|
end
|
||||||
|
if not (sep == " " and not arg_inline and nomsu:sub(1, 1) == ":") then
|
||||||
|
buff = buff .. sep
|
||||||
|
end
|
||||||
if arg_inline then
|
if arg_inline then
|
||||||
sep = " "
|
sep = " "
|
||||||
|
line_len = line_len + (1 + #nomsu)
|
||||||
else
|
else
|
||||||
|
line_len = 0
|
||||||
inline = false
|
inline = false
|
||||||
sep = "\n.."
|
sep = "\n.."
|
||||||
end
|
end
|
||||||
@ -587,7 +732,7 @@ end);]]):format(concat(buffer, "\n"))
|
|||||||
if arg_inline then
|
if arg_inline then
|
||||||
buff = buff .. "(" .. tostring(nomsu) .. ")"
|
buff = buff .. "(" .. tostring(nomsu) .. ")"
|
||||||
else
|
else
|
||||||
buff = buff .. "(..)\n " .. tostring(indent(nomsu))
|
buff = buff .. "(..)\n " .. tostring(self:indent(nomsu))
|
||||||
end
|
end
|
||||||
else
|
else
|
||||||
buff = buff .. nomsu
|
buff = buff .. nomsu
|
||||||
@ -658,6 +803,41 @@ end);]]):format(concat(buffer, "\n"))
|
|||||||
return self:error("Unknown/unimplemented thingy: " .. tostring(tree.type))
|
return self:error("Unknown/unimplemented thingy: " .. tostring(tree.type))
|
||||||
end
|
end
|
||||||
end,
|
end,
|
||||||
|
value_to_nomsu = function(self, value)
|
||||||
|
local _exp_0 = type(value)
|
||||||
|
if "nil" == _exp_0 then
|
||||||
|
return "(nil)"
|
||||||
|
elseif "bool" == _exp_0 then
|
||||||
|
return value and "(yes)" or "(no)"
|
||||||
|
elseif "number" == _exp_0 then
|
||||||
|
return repr(value)
|
||||||
|
elseif "table" == _exp_0 then
|
||||||
|
if utils.is_list(value) then
|
||||||
|
return "[" .. tostring(concat((function()
|
||||||
|
local _accum_0 = { }
|
||||||
|
local _len_0 = 1
|
||||||
|
for _index_0 = 1, #value do
|
||||||
|
local v = value[_index_0]
|
||||||
|
_accum_0[_len_0] = self:value_to_nomsu(v)
|
||||||
|
_len_0 = _len_0 + 1
|
||||||
|
end
|
||||||
|
return _accum_0
|
||||||
|
end)(), ", ")) .. "]"
|
||||||
|
else
|
||||||
|
return "(d{" .. tostring(concat((function()
|
||||||
|
local _accum_0 = { }
|
||||||
|
local _len_0 = 1
|
||||||
|
for k, v in pairs(value) do
|
||||||
|
_accum_0[_len_0] = tostring(self:value_to_nomsu(k)) .. "=" .. tostring(self:value_to_nomsu(v))
|
||||||
|
_len_0 = _len_0 + 1
|
||||||
|
end
|
||||||
|
return _accum_0
|
||||||
|
end)(), "; ")) .. "})"
|
||||||
|
end
|
||||||
|
else
|
||||||
|
return error("Unsupported value_to_nomsu type: " .. tostring(type(value)))
|
||||||
|
end
|
||||||
|
end,
|
||||||
tree_to_lua = function(self, tree)
|
tree_to_lua = function(self, tree)
|
||||||
assert(tree, "No tree provided.")
|
assert(tree, "No tree provided.")
|
||||||
if not tree.type then
|
if not tree.type then
|
||||||
@ -668,7 +848,7 @@ end);]]):format(concat(buffer, "\n"))
|
|||||||
if "File" == _exp_0 then
|
if "File" == _exp_0 then
|
||||||
return error("Should not be converting File to lua through this function.")
|
return error("Should not be converting File to lua through this function.")
|
||||||
elseif "Nomsu" == _exp_0 then
|
elseif "Nomsu" == _exp_0 then
|
||||||
return "nomsu:parse(" .. tostring(repr(tree.value.src)) .. ").value[1]", nil
|
return "nomsu:parse(" .. tostring(repr(tree.value.src)) .. ", " .. tostring(repr(CURRENT_FILE)) .. ").value[1]", nil
|
||||||
elseif "Thunk" == _exp_0 then
|
elseif "Thunk" == _exp_0 then
|
||||||
local lua_bits = { }
|
local lua_bits = { }
|
||||||
local _list_0 = tree.value
|
local _list_0 = tree.value
|
||||||
@ -688,24 +868,42 @@ local ret;
|
|||||||
return ret;
|
return ret;
|
||||||
end)]]):format(concat(lua_bits, "\n"))
|
end)]]):format(concat(lua_bits, "\n"))
|
||||||
elseif "FunctionCall" == _exp_0 then
|
elseif "FunctionCall" == _exp_0 then
|
||||||
local stub = self:get_stub(tree)
|
local def = self.defs[tree.stub]
|
||||||
local def = self.defs[stub]
|
|
||||||
if def and def.is_macro then
|
if def and def.is_macro then
|
||||||
local expr, statement = self:run_macro(tree)
|
local expr, statement = self:run_macro(tree)
|
||||||
if def.whiteset then
|
if def.whiteset then
|
||||||
if expr then
|
if expr then
|
||||||
expr = "(nomsu:assert_permission(" .. tostring(repr(stub)) .. ") and " .. tostring(expr) .. ")"
|
expr = "(nomsu:assert_permission(" .. tostring(repr(tree.stub)) .. ") and " .. tostring(expr) .. ")"
|
||||||
end
|
end
|
||||||
if statement then
|
if statement then
|
||||||
statement = "nomsu:assert_permission(" .. tostring(repr(stub)) .. ");\n" .. statement
|
statement = "nomsu:assert_permission(" .. tostring(repr(tree.stub)) .. ");\n" .. statement
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
return expr, statement
|
return expr, statement
|
||||||
end
|
end
|
||||||
local args = {
|
local args = {
|
||||||
repr(stub),
|
repr(tree.stub),
|
||||||
repr(tree.line_no)
|
repr(tree.line_no)
|
||||||
}
|
}
|
||||||
|
local arg_names, escaped_args
|
||||||
|
if def then
|
||||||
|
arg_names, escaped_args = def.arg_names, def.escaped_args
|
||||||
|
else
|
||||||
|
arg_names, escaped_args = (function()
|
||||||
|
local _accum_0 = { }
|
||||||
|
local _len_0 = 1
|
||||||
|
local _list_0 = tree.value
|
||||||
|
for _index_0 = 1, #_list_0 do
|
||||||
|
local w = _list_0[_index_0]
|
||||||
|
if w.type == "Word" then
|
||||||
|
_accum_0[_len_0] = w.value
|
||||||
|
_len_0 = _len_0 + 1
|
||||||
|
end
|
||||||
|
end
|
||||||
|
return _accum_0
|
||||||
|
end)(), { }
|
||||||
|
end
|
||||||
|
local arg_num = 1
|
||||||
local _list_0 = tree.value
|
local _list_0 = tree.value
|
||||||
for _index_0 = 1, #_list_0 do
|
for _index_0 = 1, #_list_0 do
|
||||||
local _continue_0 = false
|
local _continue_0 = false
|
||||||
@ -715,11 +913,18 @@ end)]]):format(concat(lua_bits, "\n"))
|
|||||||
_continue_0 = true
|
_continue_0 = true
|
||||||
break
|
break
|
||||||
end
|
end
|
||||||
|
if escaped_args[arg_names[arg_num]] then
|
||||||
|
arg = {
|
||||||
|
type = "Nomsu",
|
||||||
|
value = arg
|
||||||
|
}
|
||||||
|
end
|
||||||
local expr, statement = self:tree_to_lua(arg)
|
local expr, statement = self:tree_to_lua(arg)
|
||||||
if statement then
|
if statement then
|
||||||
self:error("Cannot use [[" .. tostring(arg.src) .. "]] as a function argument, since it's not an expression.")
|
self:error("Cannot use [[" .. tostring(arg.src) .. "]] as a function argument, since it's not an expression.")
|
||||||
end
|
end
|
||||||
insert(args, expr)
|
insert(args, expr)
|
||||||
|
arg_num = arg_num + 1
|
||||||
_continue_0 = true
|
_continue_0 = true
|
||||||
until true
|
until true
|
||||||
if not _continue_0 then
|
if not _continue_0 then
|
||||||
@ -883,41 +1088,38 @@ end)]]):format(concat(lua_bits, "\n"))
|
|||||||
self:error("Nothing to get stub from")
|
self:error("Nothing to get stub from")
|
||||||
end
|
end
|
||||||
if type(x) == 'string' then
|
if type(x) == 'string' then
|
||||||
local stub = x:gsub("([" .. tostring(wordbreaker) .. "]+)", " %1 "):gsub("%%%S+", "%%"):gsub("%s+", " "):gsub("^%s*", ""):gsub("%s*$", "")
|
x = x:gsub("\n%s*%.%.", " "):gsub("([" .. tostring(wordbreaker) .. "]+)", " %1 "):gsub("%s+", " ")
|
||||||
|
x = x:gsub("^%s*", ""):gsub("%s*$", "")
|
||||||
|
local stub = x:gsub("%%%S+", "%%"):gsub("\\", "")
|
||||||
local arg_names
|
local arg_names
|
||||||
do
|
do
|
||||||
local _accum_0 = { }
|
local _accum_0 = { }
|
||||||
local _len_0 = 1
|
local _len_0 = 1
|
||||||
for arg in x:gmatch("%%([^%s']*)") do
|
for arg in x:gmatch("%%([^%s]*)") do
|
||||||
_accum_0[_len_0] = arg
|
_accum_0[_len_0] = arg
|
||||||
_len_0 = _len_0 + 1
|
_len_0 = _len_0 + 1
|
||||||
end
|
end
|
||||||
arg_names = _accum_0
|
arg_names = _accum_0
|
||||||
end
|
end
|
||||||
return stub, arg_names
|
local escaped_args = utils.set((function()
|
||||||
|
local _accum_0 = { }
|
||||||
|
local _len_0 = 1
|
||||||
|
for arg in x:gmatch("\\%%([^%s]*)") do
|
||||||
|
_accum_0[_len_0] = arg
|
||||||
|
_len_0 = _len_0 + 1
|
||||||
|
end
|
||||||
|
return _accum_0
|
||||||
|
end)())
|
||||||
|
return stub, arg_names, escaped_args
|
||||||
|
end
|
||||||
|
if type(x) ~= 'table' then
|
||||||
|
self:error("Invalid type for getting stub: " .. tostring(type(x)) .. " for:\n" .. tostring(repr(x)))
|
||||||
end
|
end
|
||||||
local _exp_0 = x.type
|
local _exp_0 = x.type
|
||||||
if "String" == _exp_0 then
|
if "String" == _exp_0 then
|
||||||
return self:get_stub(x.value)
|
return self:get_stub(x.value)
|
||||||
elseif "FunctionCall" == _exp_0 then
|
elseif "FunctionCall" == _exp_0 then
|
||||||
local stub, arg_names = { }, { }, { }
|
return self:get_stub(x.src)
|
||||||
local _list_0 = x.value
|
|
||||||
for _index_0 = 1, #_list_0 do
|
|
||||||
local token = _list_0[_index_0]
|
|
||||||
local _exp_1 = token.type
|
|
||||||
if "Word" == _exp_1 then
|
|
||||||
insert(stub, token.value)
|
|
||||||
elseif "Var" == _exp_1 then
|
|
||||||
insert(stub, "%")
|
|
||||||
if arg_names then
|
|
||||||
insert(arg_names, token.value)
|
|
||||||
end
|
|
||||||
else
|
|
||||||
insert(stub, "%")
|
|
||||||
arg_names = nil
|
|
||||||
end
|
|
||||||
end
|
|
||||||
return concat(stub, " "), arg_names
|
|
||||||
else
|
else
|
||||||
return self:error("Unsupported get stub type: " .. tostring(x.type) .. " for " .. tostring(repr(x)))
|
return self:error("Unsupported get stub type: " .. tostring(x.type) .. " for " .. tostring(repr(x)))
|
||||||
end
|
end
|
||||||
@ -974,11 +1176,11 @@ end)]]):format(concat(lua_bits, "\n"))
|
|||||||
end))
|
end))
|
||||||
end,
|
end,
|
||||||
error = function(self, msg)
|
error = function(self, msg)
|
||||||
self:errorln((colored.red("ERROR!")))
|
local error_msg = colored.red("ERROR!")
|
||||||
if msg then
|
if msg then
|
||||||
self:errorln(colored.bright(colored.yellow(colored.onred(msg))))
|
error_msg = error_msg .. ("\n" .. (colored.bright(colored.yellow(colored.onred(msg)))))
|
||||||
end
|
end
|
||||||
self:errorln("Callstack:")
|
error_msg = error_msg .. "\nCallstack:"
|
||||||
local maxlen = utils.max((function()
|
local maxlen = utils.max((function()
|
||||||
local _accum_0 = { }
|
local _accum_0 = { }
|
||||||
local _len_0 = 1
|
local _len_0 = 1
|
||||||
@ -994,12 +1196,12 @@ end)]]):format(concat(lua_bits, "\n"))
|
|||||||
end)())
|
end)())
|
||||||
for i = #self.callstack, 1, -1 do
|
for i = #self.callstack, 1, -1 do
|
||||||
if self.callstack[i] ~= "#macro" then
|
if self.callstack[i] ~= "#macro" then
|
||||||
self:errorln(" " .. tostring(("%-" .. tostring(maxlen) .. "s"):format(self.callstack[i][2])) .. "| " .. tostring(self.callstack[i][1]))
|
error_msg = error_msg .. "\n " .. tostring(("%-" .. tostring(maxlen) .. "s"):format(self.callstack[i][2])) .. "| " .. tostring(self.callstack[i][1])
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
self:errorln(" <top level>")
|
error_msg = error_msg .. "\n <top level>"
|
||||||
self.callstack = { }
|
self.callstack = { }
|
||||||
return error()
|
return error(error_msg, 3)
|
||||||
end,
|
end,
|
||||||
typecheck = function(self, vars, varname, desired_type)
|
typecheck = function(self, vars, varname, desired_type)
|
||||||
local x = vars[varname]
|
local x = vars[varname]
|
||||||
@ -1009,17 +1211,19 @@ end)]]):format(concat(lua_bits, "\n"))
|
|||||||
if type(x) == 'table' and x.type == desired_type then
|
if type(x) == 'table' and x.type == desired_type then
|
||||||
return x
|
return x
|
||||||
end
|
end
|
||||||
return self:error("Invalid type for %" .. tostring(varname) .. ". Expected " .. tostring(desired_type) .. ", but got " .. tostring(x.type) .. ".")
|
return self:error("Invalid type for %" .. tostring(varname) .. ". Expected " .. tostring(desired_type) .. ", but got " .. tostring(repr(x)) .. ".")
|
||||||
end,
|
end,
|
||||||
initialize_core = function(self)
|
initialize_core = function(self)
|
||||||
local nomsu_string_as_lua
|
local nomsu_string_as_lua
|
||||||
nomsu_string_as_lua = function(self, code)
|
nomsu_string_as_lua = function(self, code, tree)
|
||||||
local concat_parts = { }
|
local concat_parts = { }
|
||||||
local _list_0 = code.value
|
local _list_0 = code.value
|
||||||
for _index_0 = 1, #_list_0 do
|
for _index_0 = 1, #_list_0 do
|
||||||
local bit = _list_0[_index_0]
|
local bit = _list_0[_index_0]
|
||||||
if type(bit) == "string" then
|
if type(bit) == "string" then
|
||||||
insert(concat_parts, bit)
|
insert(concat_parts, bit)
|
||||||
|
elseif type(bit) == "table" and bit.type == "FunctionCall" and bit.src == "__src__" then
|
||||||
|
insert(concat_parts, repr(tree.src))
|
||||||
else
|
else
|
||||||
local expr, statement = self:tree_to_lua(bit)
|
local expr, statement = self:tree_to_lua(bit)
|
||||||
if statement then
|
if statement then
|
||||||
@ -1070,12 +1274,13 @@ end)]]):format(concat(lua_bits, "\n"))
|
|||||||
self:def("run file %filename", run_file)
|
self:def("run file %filename", run_file)
|
||||||
local _require
|
local _require
|
||||||
_require = function(self, vars)
|
_require = function(self, vars)
|
||||||
if not self.loaded_files[vars.filename] then
|
local loaded = self.defs["#loaded_files"]
|
||||||
self.loaded_files[vars.filename] = run_file(self, {
|
if not loaded[vars.filename] then
|
||||||
|
loaded[vars.filename] = run_file(self, {
|
||||||
filename = vars.filename
|
filename = vars.filename
|
||||||
}) or true
|
}) or true
|
||||||
end
|
end
|
||||||
return self.loaded_files[vars.filename]
|
return loaded[vars.filename]
|
||||||
end
|
end
|
||||||
return self:def("require %filename", _require)
|
return self:def("require %filename", _require)
|
||||||
end
|
end
|
||||||
@ -1089,10 +1294,21 @@ end)]]):format(concat(lua_bits, "\n"))
|
|||||||
self.write_err = function(self, ...)
|
self.write_err = function(self, ...)
|
||||||
return io.stderr:write(...)
|
return io.stderr:write(...)
|
||||||
end
|
end
|
||||||
self.defs = setmetatable({ }, {
|
self.defs = {
|
||||||
__index = parent and parent.defs
|
["#vars"] = { },
|
||||||
})
|
["#loaded_files"] = { }
|
||||||
self.def_number = 0
|
}
|
||||||
|
if parent then
|
||||||
|
setmetatable(self.defs, {
|
||||||
|
__index = parent.defs
|
||||||
|
})
|
||||||
|
setmetatable(self.defs["#vars"], {
|
||||||
|
__index = parent["#vars"]
|
||||||
|
})
|
||||||
|
setmetatable(self.defs["#loaded_files"], {
|
||||||
|
__index = parent["#loaded_files"]
|
||||||
|
})
|
||||||
|
end
|
||||||
self.callstack = { }
|
self.callstack = { }
|
||||||
self.debug = false
|
self.debug = false
|
||||||
self.utils = utils
|
self.utils = utils
|
||||||
@ -1102,9 +1318,6 @@ end)]]):format(concat(lua_bits, "\n"))
|
|||||||
self.stringify = function(self, ...)
|
self.stringify = function(self, ...)
|
||||||
return utils.stringify(...)
|
return utils.stringify(...)
|
||||||
end
|
end
|
||||||
self.loaded_files = setmetatable({ }, {
|
|
||||||
__index = parent and parent.loaded_files
|
|
||||||
})
|
|
||||||
if not parent then
|
if not parent then
|
||||||
return self:initialize_core()
|
return self:initialize_core()
|
||||||
end
|
end
|
||||||
@ -1121,6 +1334,7 @@ end)]]):format(concat(lua_bits, "\n"))
|
|||||||
})
|
})
|
||||||
_base_0.__class = _class_0
|
_base_0.__class = _class_0
|
||||||
local self = _class_0
|
local self = _class_0
|
||||||
|
self.def_number = 0
|
||||||
self.unescape_string = function(self, str)
|
self.unescape_string = function(self, str)
|
||||||
return str:gsub("\\(.)", (function(c)
|
return str:gsub("\\(.)", (function(c)
|
||||||
return STRING_ESCAPES[c] or c
|
return STRING_ESCAPES[c] or c
|
||||||
|
222
nomsu.moon
222
nomsu.moon
@ -19,6 +19,12 @@ colors = setmetatable({}, {__index:->""})
|
|||||||
colored = setmetatable({}, {__index:(_,color)-> ((msg)-> colors[color]..msg..colors.reset)})
|
colored = setmetatable({}, {__index:(_,color)-> ((msg)-> colors[color]..msg..colors.reset)})
|
||||||
{:insert, :remove, :concat} = table
|
{:insert, :remove, :concat} = table
|
||||||
--pcall = (fn,...)-> true, fn(...)
|
--pcall = (fn,...)-> true, fn(...)
|
||||||
|
if _VERSION == "Lua 5.1"
|
||||||
|
xp = xpcall
|
||||||
|
xpcall = (f, errhandler, ...)->
|
||||||
|
args = {n:select("#", ...), ...}
|
||||||
|
return xp((...)-> f(unpack(args,1,args.n))), errhandler
|
||||||
|
--pcall = (fn, ...) -> xpcall(fn, debug.traceback, ...)
|
||||||
|
|
||||||
-- TODO:
|
-- TODO:
|
||||||
-- Maybe get GOTOs working at file scope.
|
-- Maybe get GOTOs working at file scope.
|
||||||
@ -28,7 +34,6 @@ colored = setmetatable({}, {__index:(_,color)-> ((msg)-> colors[color]..msg..col
|
|||||||
-- better scoping?
|
-- better scoping?
|
||||||
-- better error reporting
|
-- better error reporting
|
||||||
-- fix propagation of filename for error reporting
|
-- fix propagation of filename for error reporting
|
||||||
-- add line numbers of function calls
|
|
||||||
-- type checking?
|
-- type checking?
|
||||||
-- Fix compiler bug that breaks when file ends with a block comment
|
-- Fix compiler bug that breaks when file ends with a block comment
|
||||||
-- Add compiler options for optimization level (compile-fast vs. run-fast, etc.)
|
-- Add compiler options for optimization level (compile-fast vs. run-fast, etc.)
|
||||||
@ -162,7 +167,8 @@ defs =
|
|||||||
for _ in src\sub(1,pos)\gmatch("\n") do line_no += 1
|
for _ in src\sub(1,pos)\gmatch("\n") do line_no += 1
|
||||||
return pos, "#{CURRENT_FILE}:#{line_no}"
|
return pos, "#{CURRENT_FILE}:#{line_no}"
|
||||||
FunctionCall: (src, line_no, value, errors)->
|
FunctionCall: (src, line_no, value, errors)->
|
||||||
{type: "FunctionCall", :src, :line_no, :value, :errors}
|
stub = concat([(t.type == "Word" and t.value or "%") for t in *value], " ")
|
||||||
|
{type: "FunctionCall", :src, :line_no, :value, :errors, :stub}
|
||||||
error: (src,pos,errors,err_msg)->
|
error: (src,pos,errors,err_msg)->
|
||||||
line_no = 1
|
line_no = 1
|
||||||
for _ in src\sub(1,-#errors)\gmatch("\n") do line_no += 1
|
for _ in src\sub(1,-#errors)\gmatch("\n") do line_no += 1
|
||||||
@ -190,17 +196,21 @@ setmetatable(defs, {
|
|||||||
nomsu = re.compile(nomsu, defs)
|
nomsu = re.compile(nomsu, defs)
|
||||||
|
|
||||||
class NomsuCompiler
|
class NomsuCompiler
|
||||||
|
@def_number: 0
|
||||||
new:(parent)=>
|
new:(parent)=>
|
||||||
@write = (...)=> io.write(...)
|
@write = (...)=> io.write(...)
|
||||||
@write_err = (...)=> io.stderr\write(...)
|
@write_err = (...)=> io.stderr\write(...)
|
||||||
@defs = setmetatable({}, {__index:parent and parent.defs})
|
-- Use # to prevent someone from defining a function that has a namespace collision.
|
||||||
@def_number = 0
|
@defs = {["#vars"]:{}, ["#loaded_files"]:{}}
|
||||||
|
if parent
|
||||||
|
setmetatable(@defs, {__index:parent.defs})
|
||||||
|
setmetatable(@defs["#vars"], {__index:parent["#vars"]})
|
||||||
|
setmetatable(@defs["#loaded_files"], {__index:parent["#loaded_files"]})
|
||||||
@callstack = {}
|
@callstack = {}
|
||||||
@debug = false
|
@debug = false
|
||||||
@utils = utils
|
@utils = utils
|
||||||
@repr = (...)=> repr(...)
|
@repr = (...)=> repr(...)
|
||||||
@stringify = (...)=> utils.stringify(...)
|
@stringify = (...)=> utils.stringify(...)
|
||||||
@loaded_files = setmetatable({}, {__index:parent and parent.loaded_files})
|
|
||||||
if not parent
|
if not parent
|
||||||
@initialize_core!
|
@initialize_core!
|
||||||
|
|
||||||
@ -219,9 +229,12 @@ class NomsuCompiler
|
|||||||
signature = @get_stubs signature
|
signature = @get_stubs signature
|
||||||
assert type(thunk) == 'function', "Bad thunk: #{repr thunk}"
|
assert type(thunk) == 'function', "Bad thunk: #{repr thunk}"
|
||||||
canonical_args = nil
|
canonical_args = nil
|
||||||
|
canonical_escaped_args = nil
|
||||||
aliases = {}
|
aliases = {}
|
||||||
@def_number += 1
|
@@def_number += 1
|
||||||
for {stub, arg_names} in *signature
|
def = {:thunk, :src, :is_macro, aliases:{}, def_number:@@def_number, defs:@defs}
|
||||||
|
where_defs_go = ((getmetatable(@defs) or {}).__newindex) or @defs
|
||||||
|
for {stub, arg_names, escaped_args} in *signature
|
||||||
assert stub, "NO STUB FOUND: #{repr signature}"
|
assert stub, "NO STUB FOUND: #{repr signature}"
|
||||||
if @debug then @writeln "#{colored.bright "DEFINING RULE:"} #{colored.underscore colored.magenta repr(stub)} #{colored.bright "WITH ARGS"} #{colored.dim repr(arg_names)}"
|
if @debug then @writeln "#{colored.bright "DEFINING RULE:"} #{colored.underscore colored.magenta repr(stub)} #{colored.bright "WITH ARGS"} #{colored.dim repr(arg_names)}"
|
||||||
for i=1,#arg_names-1 do for j=i+1,#arg_names
|
for i=1,#arg_names-1 do for j=i+1,#arg_names
|
||||||
@ -229,22 +242,58 @@ class NomsuCompiler
|
|||||||
if canonical_args
|
if canonical_args
|
||||||
assert utils.equivalent(utils.set(arg_names), canonical_args), "Mismatched args"
|
assert utils.equivalent(utils.set(arg_names), canonical_args), "Mismatched args"
|
||||||
else canonical_args = utils.set(arg_names)
|
else canonical_args = utils.set(arg_names)
|
||||||
insert aliases, stub
|
if canonical_escaped_args
|
||||||
@defs[stub] = {:thunk, :stub, :arg_names, :src, :is_macro, :aliases, def_number:@def_number}
|
assert utils.equivalent(escaped_args, canonical_escaped_args), "Mismatched escaped args"
|
||||||
|
else
|
||||||
|
canonical_escaped_args = escaped_args
|
||||||
|
def.escaped_args = escaped_args
|
||||||
|
insert def.aliases, stub
|
||||||
|
stub_def = setmetatable({:stub, :arg_names, :escaped_args}, {__index:def})
|
||||||
|
rawset(where_defs_go, stub, stub_def)
|
||||||
|
|
||||||
defmacro: (signature, thunk, src)=>
|
defmacro: (signature, thunk, src)=>
|
||||||
@def(signature, thunk, src, true)
|
@def(signature, thunk, src, true)
|
||||||
|
|
||||||
serialize_defs: (after=0)=>
|
scoped: (thunk)=>
|
||||||
defs = {}
|
old_defs = @defs
|
||||||
for _, def in pairs(@defs)
|
@defs = setmetatable({}, {__index:old_defs})
|
||||||
defs[def.def_number] = def.src or ""
|
ok, ret1, ret2 = pcall thunk, @
|
||||||
keys = utils.keys(defs)
|
@defs = old_defs
|
||||||
|
if not ok then @error(ret1)
|
||||||
|
return ret1, ret2
|
||||||
|
|
||||||
|
serialize_defs: (scope=nil, after=0)=>
|
||||||
|
scope or= @defs
|
||||||
|
defs_by_num = {}
|
||||||
|
for stub, def in pairs(scope)
|
||||||
|
if def and stub\sub(1,1) != "#"
|
||||||
|
defs_by_num[def.def_number] = def
|
||||||
|
keys = [k for k,v in pairs(defs_by_num)]
|
||||||
table.sort(keys)
|
table.sort(keys)
|
||||||
|
|
||||||
buff = {}
|
buff = {}
|
||||||
for i in *keys
|
k_i = 1
|
||||||
if i > after and #defs[i] > 0
|
_using = nil
|
||||||
insert buff, defs[i]
|
_using_do = {}
|
||||||
|
for k_i,i in ipairs(keys)
|
||||||
|
if i <= after then continue
|
||||||
|
def = defs_by_num[i]
|
||||||
|
if def.defs == scope
|
||||||
|
if def.src
|
||||||
|
insert buff, def.src
|
||||||
|
continue
|
||||||
|
if _using == def.defs
|
||||||
|
if def.src
|
||||||
|
insert _using_do, def.src
|
||||||
|
else
|
||||||
|
_using = def.defs
|
||||||
|
_using_do = {def.src}
|
||||||
|
if k_i == #keys or defs_by_num[keys[k_i+1]].defs != _using
|
||||||
|
insert buff, "using:\n #{@indent @serialize_defs(_using)}\n..do:\n #{@indent concat(_using_do, "\n")}"
|
||||||
|
|
||||||
|
for k,v in pairs(scope["#vars"] or {})
|
||||||
|
insert buff, "<@#{k}> = #{@value_to_nomsu v}"
|
||||||
|
|
||||||
return concat buff, "\n"
|
return concat buff, "\n"
|
||||||
|
|
||||||
call: (stub,line_no,...)=>
|
call: (stub,line_no,...)=>
|
||||||
@ -254,7 +303,7 @@ class NomsuCompiler
|
|||||||
if def and def.is_macro and @callstack[#@callstack] != "#macro"
|
if def and def.is_macro and @callstack[#@callstack] != "#macro"
|
||||||
@error "Attempt to call macro at runtime: #{stub}\nThis can be caused by using a macro in a function that is defined before the macro."
|
@error "Attempt to call macro at runtime: #{stub}\nThis can be caused by using a macro in a function that is defined before the macro."
|
||||||
insert @callstack, {stub, line_no}
|
insert @callstack, {stub, line_no}
|
||||||
if def == nil
|
unless def
|
||||||
@error "Attempt to call undefined function: #{stub}"
|
@error "Attempt to call undefined function: #{stub}"
|
||||||
unless def.is_macro
|
unless def.is_macro
|
||||||
@assert_permission(stub)
|
@assert_permission(stub)
|
||||||
@ -263,22 +312,41 @@ class NomsuCompiler
|
|||||||
if @debug
|
if @debug
|
||||||
@write "#{colored.bright "CALLING"} #{colored.magenta(colored.underscore stub)} "
|
@write "#{colored.bright "CALLING"} #{colored.magenta(colored.underscore stub)} "
|
||||||
@writeln "#{colored.bright "WITH ARGS:"} #{colored.dim repr(args)}"
|
@writeln "#{colored.bright "WITH ARGS:"} #{colored.dim repr(args)}"
|
||||||
-- TODO: optimize, but still allow multiple return values?
|
old_defs, @defs = @defs, def.defs
|
||||||
rets = {thunk(self,args)}
|
rets = {thunk(self,args)}
|
||||||
|
@defs = old_defs
|
||||||
remove @callstack
|
remove @callstack
|
||||||
return unpack(rets)
|
return unpack(rets)
|
||||||
|
|
||||||
run_macro: (tree)=>
|
run_macro: (tree)=>
|
||||||
stub = @get_stub tree
|
|
||||||
args = [arg for arg in *tree.value when arg.type != "Word"]
|
args = [arg for arg in *tree.value when arg.type != "Word"]
|
||||||
if @debug
|
if @debug
|
||||||
@write "#{colored.bright "RUNNING MACRO"} #{colored.underscore colored.magenta(stub)} "
|
@write "#{colored.bright "RUNNING MACRO"} #{colored.underscore colored.magenta(tree.stub)} "
|
||||||
@writeln "#{colored.bright "WITH ARGS:"} #{colored.dim repr args}"
|
@writeln "#{colored.bright "WITH ARGS:"} #{colored.dim repr args}"
|
||||||
insert @callstack, "#macro"
|
insert @callstack, "#macro"
|
||||||
expr, statement = @call(stub, tree.line_no, unpack(args))
|
expr, statement = @call(tree.stub, tree.line_no, unpack(args))
|
||||||
remove @callstack
|
remove @callstack
|
||||||
return expr, statement
|
return expr, statement
|
||||||
|
|
||||||
|
dedent: (code)=>
|
||||||
|
unless code\find("\n")
|
||||||
|
return code
|
||||||
|
spaces, indent_spaces = math.huge, math.huge
|
||||||
|
for line in code\gmatch("\n([^\n]*)")
|
||||||
|
if line\match("^%s*#.*")
|
||||||
|
continue
|
||||||
|
elseif s = line\match("^(%s*)%.%..*")
|
||||||
|
spaces = math.min(spaces, #s)
|
||||||
|
elseif s = line\match("^(%s*)%S.*")
|
||||||
|
indent_spaces = math.min(indent_spaces, #s)
|
||||||
|
if spaces != math.huge and spaces < indent_spaces
|
||||||
|
return (code\gsub("\n"..(" ")\rep(spaces), "\n"))
|
||||||
|
else
|
||||||
|
return (code\gsub("\n"..(" ")\rep(indent_spaces), "\n "))
|
||||||
|
|
||||||
|
indent: (code)=>
|
||||||
|
(code\gsub("\n","\n "))
|
||||||
|
|
||||||
assert_permission: (stub)=>
|
assert_permission: (stub)=>
|
||||||
fn_def = @defs[stub]
|
fn_def = @defs[stub]
|
||||||
unless fn_def
|
unless fn_def
|
||||||
@ -343,7 +411,7 @@ class NomsuCompiler
|
|||||||
ok,expr,statements = pcall(@tree_to_lua, self, statement)
|
ok,expr,statements = pcall(@tree_to_lua, self, statement)
|
||||||
if not ok
|
if not ok
|
||||||
@errorln "#{colored.red "Error occurred in statement:"}\n#{colored.bright colored.yellow statement.src}"
|
@errorln "#{colored.red "Error occurred in statement:"}\n#{colored.bright colored.yellow statement.src}"
|
||||||
@error(expr)
|
error(expr)
|
||||||
code_for_statement = ([[
|
code_for_statement = ([[
|
||||||
return (function(nomsu, vars)
|
return (function(nomsu, vars)
|
||||||
%s
|
%s
|
||||||
@ -365,7 +433,7 @@ end);]])\format(statements or "", expr or "ret")
|
|||||||
if not ok
|
if not ok
|
||||||
@errorln "#{colored.red "Error occurred in statement:"}\n#{colored.yellow statement.src}"
|
@errorln "#{colored.red "Error occurred in statement:"}\n#{colored.yellow statement.src}"
|
||||||
@errorln debug.traceback!
|
@errorln debug.traceback!
|
||||||
@error(ret)
|
error(ret)
|
||||||
if statements
|
if statements
|
||||||
insert buffer, statements
|
insert buffer, statements
|
||||||
if expr
|
if expr
|
||||||
@ -392,8 +460,6 @@ end);]])\format(concat(buffer, "\n"))
|
|||||||
|
|
||||||
tree_to_nomsu: (tree, force_inline=false)=>
|
tree_to_nomsu: (tree, force_inline=false)=>
|
||||||
-- Return <nomsu code>, <is safe for inline use>
|
-- Return <nomsu code>, <is safe for inline use>
|
||||||
indent = (s)->
|
|
||||||
s\gsub("\n","\n ")
|
|
||||||
assert tree, "No tree provided."
|
assert tree, "No tree provided."
|
||||||
if not tree.type
|
if not tree.type
|
||||||
@errorln debug.traceback()
|
@errorln debug.traceback()
|
||||||
@ -410,27 +476,31 @@ end);]])\format(concat(buffer, "\n"))
|
|||||||
if force_inline
|
if force_inline
|
||||||
return "{#{concat([@tree_to_nomsu(v, true) for v in *tree.value], "; ")}", true
|
return "{#{concat([@tree_to_nomsu(v, true) for v in *tree.value], "; ")}", true
|
||||||
else
|
else
|
||||||
return ":"..indent("\n"..concat([@tree_to_nomsu v for v in *tree.value], "\n")), false
|
return ":"..@indent("\n"..concat([@tree_to_nomsu v for v in *tree.value], "\n")), false
|
||||||
|
|
||||||
when "FunctionCall"
|
when "FunctionCall"
|
||||||
buff = ""
|
buff = ""
|
||||||
sep = ""
|
sep = ""
|
||||||
inline = true
|
inline = true
|
||||||
do_arg = (arg)->
|
line_len = 0
|
||||||
|
|
||||||
for arg in *tree.value
|
for arg in *tree.value
|
||||||
nomsu, arg_inline = @tree_to_nomsu(arg, force_inline)
|
nomsu, arg_inline = @tree_to_nomsu(arg, force_inline)
|
||||||
buff ..= sep
|
if sep == " " and line_len + #nomsu > 80
|
||||||
|
sep = "\n.."
|
||||||
|
unless sep == " " and not arg_inline and nomsu\sub(1,1) == ":"
|
||||||
|
buff ..= sep
|
||||||
if arg_inline
|
if arg_inline
|
||||||
sep = " "
|
sep = " "
|
||||||
|
line_len += 1 + #nomsu
|
||||||
else
|
else
|
||||||
|
line_len = 0
|
||||||
inline = false
|
inline = false
|
||||||
sep = "\n.."
|
sep = "\n.."
|
||||||
if arg.type == 'FunctionCall'
|
if arg.type == 'FunctionCall'
|
||||||
if arg_inline
|
if arg_inline
|
||||||
buff ..= "(#{nomsu})"
|
buff ..= "(#{nomsu})"
|
||||||
else
|
else
|
||||||
buff ..= "(..)\n #{indent nomsu}"
|
buff ..= "(..)\n #{@indent nomsu}"
|
||||||
else
|
else
|
||||||
buff ..= nomsu
|
buff ..= nomsu
|
||||||
return buff, inline
|
return buff, inline
|
||||||
@ -491,6 +561,22 @@ end);]])\format(concat(buffer, "\n"))
|
|||||||
else
|
else
|
||||||
@error("Unknown/unimplemented thingy: #{tree.type}")
|
@error("Unknown/unimplemented thingy: #{tree.type}")
|
||||||
|
|
||||||
|
value_to_nomsu: (value)=>
|
||||||
|
switch type(value)
|
||||||
|
when "nil"
|
||||||
|
return "(nil)"
|
||||||
|
when "bool"
|
||||||
|
return value and "(yes)" or "(no)"
|
||||||
|
when "number"
|
||||||
|
return repr(value)
|
||||||
|
when "table"
|
||||||
|
if utils.is_list(value)
|
||||||
|
return "[#{concat [@value_to_nomsu(v) for v in *value], ", "}]"
|
||||||
|
else
|
||||||
|
return "(d{#{concat ["#{@value_to_nomsu(k)}=#{@value_to_nomsu(v)}" for k,v in pairs(value)], "; "}})"
|
||||||
|
else
|
||||||
|
error("Unsupported value_to_nomsu type: #{type(value)}")
|
||||||
|
|
||||||
tree_to_lua: (tree)=>
|
tree_to_lua: (tree)=>
|
||||||
-- Return <lua code for value>, <additional lua code>
|
-- Return <lua code for value>, <additional lua code>
|
||||||
assert tree, "No tree provided."
|
assert tree, "No tree provided."
|
||||||
@ -502,7 +588,7 @@ end);]])\format(concat(buffer, "\n"))
|
|||||||
error("Should not be converting File to lua through this function.")
|
error("Should not be converting File to lua through this function.")
|
||||||
|
|
||||||
when "Nomsu"
|
when "Nomsu"
|
||||||
return "nomsu:parse(#{repr tree.value.src}).value[1]", nil
|
return "nomsu:parse(#{repr tree.value.src}, #{repr CURRENT_FILE}).value[1]", nil
|
||||||
|
|
||||||
when "Thunk"
|
when "Thunk"
|
||||||
lua_bits = {}
|
lua_bits = {}
|
||||||
@ -518,23 +604,31 @@ return ret;
|
|||||||
end)]])\format(concat(lua_bits, "\n"))
|
end)]])\format(concat(lua_bits, "\n"))
|
||||||
|
|
||||||
when "FunctionCall"
|
when "FunctionCall"
|
||||||
stub = @get_stub(tree)
|
def = @defs[tree.stub]
|
||||||
def = @defs[stub]
|
|
||||||
if def and def.is_macro
|
if def and def.is_macro
|
||||||
expr, statement = @run_macro(tree)
|
expr, statement = @run_macro(tree)
|
||||||
if def.whiteset
|
if def.whiteset
|
||||||
if expr
|
if expr
|
||||||
expr = "(nomsu:assert_permission(#{repr stub}) and #{expr})"
|
expr = "(nomsu:assert_permission(#{repr tree.stub}) and #{expr})"
|
||||||
if statement
|
if statement
|
||||||
statement = "nomsu:assert_permission(#{repr stub});\n"..statement
|
statement = "nomsu:assert_permission(#{repr tree.stub});\n"..statement
|
||||||
return expr, statement
|
return expr, statement
|
||||||
args = {repr(stub), repr(tree.line_no)}
|
args = {repr(tree.stub), repr(tree.line_no)}
|
||||||
|
local arg_names, escaped_args
|
||||||
|
if def
|
||||||
|
arg_names, escaped_args = def.arg_names, def.escaped_args
|
||||||
|
else
|
||||||
|
arg_names, escaped_args = [w.value for w in *tree.value when w.type == "Word"], {}
|
||||||
|
arg_num = 1
|
||||||
for arg in *tree.value
|
for arg in *tree.value
|
||||||
if arg.type == 'Word' then continue
|
if arg.type == 'Word' then continue
|
||||||
|
if escaped_args[arg_names[arg_num]]
|
||||||
|
arg = {type:"Nomsu", value:arg}
|
||||||
expr,statement = @tree_to_lua arg
|
expr,statement = @tree_to_lua arg
|
||||||
if statement
|
if statement
|
||||||
@error "Cannot use [[#{arg.src}]] as a function argument, since it's not an expression."
|
@error "Cannot use [[#{arg.src}]] as a function argument, since it's not an expression."
|
||||||
insert args, expr
|
insert args, expr
|
||||||
|
arg_num += 1
|
||||||
return @@comma_separated_items("nomsu:call(", args, ")"), nil
|
return @@comma_separated_items("nomsu:call(", args, ")"), nil
|
||||||
|
|
||||||
when "String"
|
when "String"
|
||||||
@ -658,27 +752,22 @@ end)]])\format(concat(lua_bits, "\n"))
|
|||||||
get_stub: (x)=>
|
get_stub: (x)=>
|
||||||
if not x
|
if not x
|
||||||
@error "Nothing to get stub from"
|
@error "Nothing to get stub from"
|
||||||
-- Returns a single stub ("say %"), and list of arg names ({"msg"}) from a single rule def
|
-- Returns a single stub ("say %"), list of arg names ({"msg"}), and set of arg
|
||||||
|
-- names that should not be evaluated from a single rule def
|
||||||
-- (e.g. "say %msg") or function call (e.g. FunctionCall({Word("say"), Var("msg")))
|
-- (e.g. "say %msg") or function call (e.g. FunctionCall({Word("say"), Var("msg")))
|
||||||
if type(x) == 'string'
|
if type(x) == 'string'
|
||||||
stub = x\gsub("([#{wordbreaker}]+)"," %1 ")\gsub("%%%S+","%%")\gsub("%s+"," ")\gsub("^%s*","")\gsub("%s*$","")
|
-- Standardize format to stuff separated by spaces
|
||||||
arg_names = [arg for arg in x\gmatch("%%([^%s']*)")]
|
x = x\gsub("\n%s*%.%.", " ")\gsub("([#{wordbreaker}]+)", " %1 ")\gsub("%s+"," ")
|
||||||
return stub, arg_names
|
x = x\gsub("^%s*","")\gsub("%s*$","")
|
||||||
|
stub = x\gsub("%%%S+","%%")\gsub("\\","")
|
||||||
|
arg_names = [arg for arg in x\gmatch("%%([^%s]*)")]
|
||||||
|
escaped_args = utils.set [arg for arg in x\gmatch("\\%%([^%s]*)")]
|
||||||
|
return stub, arg_names, escaped_args
|
||||||
|
if type(x) != 'table'
|
||||||
|
@error "Invalid type for getting stub: #{type(x)} for:\n#{repr x}"
|
||||||
switch x.type
|
switch x.type
|
||||||
when "String" then return @get_stub(x.value)
|
when "String" then return @get_stub(x.value)
|
||||||
when "FunctionCall"
|
when "FunctionCall" then return @get_stub(x.src)
|
||||||
stub, arg_names = {}, {}, {}
|
|
||||||
for token in *x.value
|
|
||||||
switch token.type
|
|
||||||
when "Word"
|
|
||||||
insert stub, token.value
|
|
||||||
when "Var"
|
|
||||||
insert stub, "%"
|
|
||||||
if arg_names then insert arg_names, token.value
|
|
||||||
else
|
|
||||||
insert stub, "%"
|
|
||||||
arg_names = nil
|
|
||||||
return concat(stub," "), arg_names
|
|
||||||
else @error "Unsupported get stub type: #{x.type} for #{repr x}"
|
else @error "Unsupported get stub type: #{x.type} for #{repr x}"
|
||||||
|
|
||||||
get_stubs: (x)=>
|
get_stubs: (x)=>
|
||||||
@ -699,31 +788,33 @@ end)]])\format(concat(lua_bits, "\n"))
|
|||||||
if verboten == "_" then "__" else ("_%x")\format(verboten\byte!))
|
if verboten == "_" then "__" else ("_%x")\format(verboten\byte!))
|
||||||
|
|
||||||
error: (msg)=>
|
error: (msg)=>
|
||||||
@errorln (colored.red "ERROR!")
|
error_msg = colored.red "ERROR!"
|
||||||
if msg
|
if msg
|
||||||
@errorln(colored.bright colored.yellow colored.onred msg)
|
error_msg ..= "\n" .. (colored.bright colored.yellow colored.onred msg)
|
||||||
@errorln("Callstack:")
|
error_msg ..= "\nCallstack:"
|
||||||
maxlen = utils.max([#c[2] for c in *@callstack when c != "#macro"])
|
maxlen = utils.max([#c[2] for c in *@callstack when c != "#macro"])
|
||||||
for i=#@callstack,1,-1
|
for i=#@callstack,1,-1
|
||||||
if @callstack[i] != "#macro"
|
if @callstack[i] != "#macro"
|
||||||
@errorln " #{"%-#{maxlen}s"\format @callstack[i][2]}| #{@callstack[i][1]}"
|
error_msg ..= "\n #{"%-#{maxlen}s"\format @callstack[i][2]}| #{@callstack[i][1]}"
|
||||||
@errorln " <top level>"
|
error_msg ..= "\n <top level>"
|
||||||
@callstack = {}
|
@callstack = {}
|
||||||
error!
|
error error_msg, 3
|
||||||
|
|
||||||
typecheck: (vars, varname, desired_type)=>
|
typecheck: (vars, varname, desired_type)=>
|
||||||
x = vars[varname]
|
x = vars[varname]
|
||||||
if type(x) == desired_type then return x
|
if type(x) == desired_type then return x
|
||||||
if type(x) == 'table' and x.type == desired_type then return x
|
if type(x) == 'table' and x.type == desired_type then return x
|
||||||
@error "Invalid type for %#{varname}. Expected #{desired_type}, but got #{x.type}."
|
@error "Invalid type for %#{varname}. Expected #{desired_type}, but got #{repr x}."
|
||||||
|
|
||||||
initialize_core: =>
|
initialize_core: =>
|
||||||
-- Sets up some core functionality
|
-- Sets up some core functionality
|
||||||
nomsu_string_as_lua = (code)=>
|
nomsu_string_as_lua = (code, tree)=>
|
||||||
concat_parts = {}
|
concat_parts = {}
|
||||||
for bit in *code.value
|
for bit in *code.value
|
||||||
if type(bit) == "string"
|
if type(bit) == "string"
|
||||||
insert concat_parts, bit
|
insert concat_parts, bit
|
||||||
|
elseif type(bit) == "table" and bit.type == "FunctionCall" and bit.src == "__src__"
|
||||||
|
insert concat_parts, repr(tree.src)
|
||||||
else
|
else
|
||||||
expr, statement = @tree_to_lua bit
|
expr, statement = @tree_to_lua bit
|
||||||
if statement
|
if statement
|
||||||
@ -763,9 +854,10 @@ end)]])\format(concat(lua_bits, "\n"))
|
|||||||
@def "run file %filename", run_file
|
@def "run file %filename", run_file
|
||||||
|
|
||||||
_require = (vars)=>
|
_require = (vars)=>
|
||||||
if not @loaded_files[vars.filename]
|
loaded = @defs["#loaded_files"]
|
||||||
@loaded_files[vars.filename] = run_file(self, {filename:vars.filename}) or true
|
if not loaded[vars.filename]
|
||||||
return @loaded_files[vars.filename]
|
loaded[vars.filename] = run_file(self, {filename:vars.filename}) or true
|
||||||
|
return loaded[vars.filename]
|
||||||
@def "require %filename", _require
|
@def "require %filename", _require
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user