Changed generated code to be less verbose for function and macro defs
(using "foo %" syntax instead of {type="FunctionCall", ...} literals).
This commit is contained in:
parent
1083273b9f
commit
7435b61380
@ -5,7 +5,7 @@
|
|||||||
# Rule to make rules:
|
# Rule to make rules:
|
||||||
lua code ".."
|
lua code ".."
|
||||||
|nomsu:defmacro("rule %signature = %body", (function(nomsu, vars)
|
|nomsu:defmacro("rule %signature = %body", (function(nomsu, vars)
|
||||||
| local signature = nomsu:typecheck(vars, "signature", "List").value;
|
| local signature = nomsu:get_stubs(nomsu:typecheck(vars, "signature", "List").value);
|
||||||
| local body = nomsu:typecheck(vars, "body", "Thunk");
|
| local body = nomsu:typecheck(vars, "body", "Thunk");
|
||||||
| return ([[
|
| return ([[
|
||||||
|nomsu:def(%s, %s, %s)
|
|nomsu:def(%s, %s, %s)
|
||||||
@ -15,7 +15,7 @@ lua code ".."
|
|||||||
# Rule to make nomsu macros:
|
# Rule to make nomsu macros:
|
||||||
rule [escaped parse %shorthand as %longhand] =:
|
rule [escaped parse %shorthand as %longhand] =:
|
||||||
lua code ".."
|
lua code ".."
|
||||||
|local aliases = nomsu:typecheck(vars, "shorthand", "List").value;
|
|local aliases = nomsu:get_stubs(nomsu:typecheck(vars, "shorthand", "List").value);
|
||||||
|if #vars.longhand.value ~= 1 then;
|
|if #vars.longhand.value ~= 1 then;
|
||||||
| nomsu:error("Expected only 1 line to parse to, but got "..tostring(#vars.longhand.value));
|
| nomsu:error("Expected only 1 line to parse to, but got "..tostring(#vars.longhand.value));
|
||||||
|end;
|
|end;
|
||||||
@ -30,13 +30,13 @@ escaped parse \[parse %shorthand as %longhand] as \: escaped parse \%shorthand a
|
|||||||
# Rule to make lua macros:
|
# Rule to make lua macros:
|
||||||
rule [escaped compile %macro_def to %body] =:
|
rule [escaped compile %macro_def to %body] =:
|
||||||
lua code ".."
|
lua code ".."
|
||||||
|local aliases = nomsu:typecheck(vars, "macro_def", "List").value;
|
|local aliases = nomsu:get_stubs(nomsu:typecheck(vars, "macro_def", "List").value);
|
||||||
|local body = nomsu:typecheck(vars, "body", "Thunk");
|
|local body = nomsu:typecheck(vars, "body", "Thunk");
|
||||||
|local thunk = nomsu:tree_to_value(body);
|
|local thunk = nomsu:tree_to_value(body);
|
||||||
|nomsu:defmacro(aliases, thunk, body.src);
|
|nomsu:defmacro(aliases, thunk, body.src);
|
||||||
rule [escaped compile %macro_def to code %body] =:
|
rule [escaped compile %macro_def to code %body] =:
|
||||||
lua code ".."
|
lua code ".."
|
||||||
|local aliases = nomsu:typecheck(vars, "macro_def", "List").value;
|
|local aliases = nomsu:get_stubs(nomsu:typecheck(vars, "macro_def", "List").value);
|
||||||
|local body = nomsu:typecheck(vars, "body", "Thunk");
|
|local body = nomsu:typecheck(vars, "body", "Thunk");
|
||||||
|local thunk = nomsu:tree_to_value(body);
|
|local thunk = nomsu:tree_to_value(body);
|
||||||
|local thunk_wrapper = function(nomsu, vars) return nil, thunk(nomsu, vars); end;
|
|local thunk_wrapper = function(nomsu, vars) return nil, thunk(nomsu, vars); end;
|
||||||
|
107
nomsu.lua
107
nomsu.lua
@ -54,7 +54,7 @@ check_nodent = function(subject, end_pos, spaces)
|
|||||||
return end_pos
|
return end_pos
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
local nomsu = [=[ file <- ({ {| shebang?
|
local nomsu = [=[ file <- ({{| shebang?
|
||||||
(ignored_line %nl)*
|
(ignored_line %nl)*
|
||||||
statements (nodent statements)*
|
statements (nodent statements)*
|
||||||
(%nl ignored_line)* %nl?
|
(%nl ignored_line)* %nl?
|
||||||
@ -77,9 +77,9 @@ local nomsu = [=[ file <- ({ {| shebang?
|
|||||||
(dedent / (({.+} ("" -> "Error while parsing thunk")) => error))
|
(dedent / (({.+} ("" -> "Error while parsing thunk")) => error))
|
||||||
|} }) -> Thunk
|
|} }) -> Thunk
|
||||||
|
|
||||||
inline_nomsu <- ({ ("\" inline_expression) }) -> Nomsu
|
inline_nomsu <- ({("\" inline_expression) }) -> Nomsu
|
||||||
eol_nomsu <- ({ ("\" noeol_expression) }) -> Nomsu
|
eol_nomsu <- ({("\" noeol_expression) }) -> Nomsu
|
||||||
indented_nomsu <- ({ ("\" expression) }) -> Nomsu
|
indented_nomsu <- ({("\" expression) }) -> Nomsu
|
||||||
|
|
||||||
inline_expression <- number / variable / inline_string / inline_list / inline_nomsu
|
inline_expression <- number / variable / inline_string / inline_list / inline_nomsu
|
||||||
/ inline_thunk / ("(" inline_statement ")")
|
/ inline_thunk / ("(" inline_statement ")")
|
||||||
@ -91,13 +91,13 @@ local nomsu = [=[ file <- ({ {| shebang?
|
|||||||
expression <- eol_thunk / eol_nomsu / noeol_expression
|
expression <- eol_thunk / eol_nomsu / noeol_expression
|
||||||
|
|
||||||
-- Function calls need at least one word in them
|
-- Function calls need at least one word in them
|
||||||
inline_functioncall <- ({ {|
|
inline_functioncall <- ({(''=>line_no) {|
|
||||||
(inline_expression tok_gap)* word (tok_gap (inline_expression / word))*
|
(inline_expression tok_gap)* word (tok_gap (inline_expression / word))*
|
||||||
|} }) -> FunctionCall
|
|} }) -> FunctionCall
|
||||||
noeol_functioncall <- ({ {|
|
noeol_functioncall <- ({(''=>line_no) {|
|
||||||
(noeol_expression tok_gap)* word (tok_gap (noeol_expression / word))*
|
(noeol_expression tok_gap)* word (tok_gap (noeol_expression / word))*
|
||||||
|} }) -> FunctionCall
|
|} }) -> FunctionCall
|
||||||
functioncall <- ({ {|
|
functioncall <- ({(''=>line_no) {|
|
||||||
(expression (dotdot / tok_gap))* word ((dotdot / tok_gap) (expression / word))*
|
(expression (dotdot / tok_gap))* word ((dotdot / tok_gap) (expression / word))*
|
||||||
|} }) -> FunctionCall
|
|} }) -> FunctionCall
|
||||||
|
|
||||||
@ -145,6 +145,7 @@ local nomsu = [=[ file <- ({ {| shebang?
|
|||||||
semicolon <- %ws? ";" %ws?
|
semicolon <- %ws? ";" %ws?
|
||||||
dotdot <- nodent ".." %ws?
|
dotdot <- nodent ".." %ws?
|
||||||
]=]
|
]=]
|
||||||
|
local CURRENT_FILE = nil
|
||||||
local whitespace = S(" \t") ^ 1
|
local whitespace = S(" \t") ^ 1
|
||||||
local defs = {
|
local defs = {
|
||||||
ws = whitespace,
|
ws = whitespace,
|
||||||
@ -155,6 +156,22 @@ local defs = {
|
|||||||
nodented = Cmt(S(" \t") ^ 0 * (#(P(1) - S(" \t\n") + (-P(1)))), check_nodent),
|
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),
|
dedented = Cmt(S(" \t") ^ 0 * (#(P(1) - S(" \t\n") + (-P(1)))), check_dedent),
|
||||||
prev_edge = B(S(" \t\n.,:;}])\"\\")),
|
prev_edge = B(S(" \t\n.,:;}])\"\\")),
|
||||||
|
line_no = function(src, pos)
|
||||||
|
local line_no = 1
|
||||||
|
for _ in src:sub(1, pos):gmatch("\n") do
|
||||||
|
line_no = line_no + 1
|
||||||
|
end
|
||||||
|
return pos, tostring(CURRENT_FILE) .. ":" .. tostring(line_no)
|
||||||
|
end,
|
||||||
|
FunctionCall = function(src, line_no, value, errors)
|
||||||
|
return {
|
||||||
|
type = "FunctionCall",
|
||||||
|
src = src,
|
||||||
|
line_no = line_no,
|
||||||
|
value = value,
|
||||||
|
errors = errors
|
||||||
|
}
|
||||||
|
end,
|
||||||
error = function(src, pos, errors, err_msg)
|
error = function(src, pos, errors, err_msg)
|
||||||
local line_no = 1
|
local line_no = 1
|
||||||
for _ in src:sub(1, -#errors):gmatch("\n") do
|
for _ in src:sub(1, -#errors):gmatch("\n") do
|
||||||
@ -175,7 +192,7 @@ local defs = {
|
|||||||
local prev_line, err_line, next_line
|
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)
|
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) .. "^"
|
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")
|
return error("\n" .. tostring(err_msg or "Parse error") .. " in " .. tostring(CURRENT_FILE) .. " on line " .. tostring(line_no) .. ":\n\n" .. tostring(prev_line) .. "\n" .. tostring(err_line) .. "\n" .. tostring(pointer) .. "\n" .. tostring(next_line) .. "\n")
|
||||||
end
|
end
|
||||||
}
|
}
|
||||||
setmetatable(defs, {
|
setmetatable(defs, {
|
||||||
@ -213,12 +230,18 @@ do
|
|||||||
if is_macro == nil then
|
if is_macro == nil then
|
||||||
is_macro = false
|
is_macro = false
|
||||||
end
|
end
|
||||||
|
if type(signature) == 'string' then
|
||||||
|
signature = self:get_stubs({
|
||||||
|
signature
|
||||||
|
})
|
||||||
|
elseif type(signature) == 'table' and type(signature[1]) == 'string' then
|
||||||
|
signature = self:get_stubs(signature)
|
||||||
|
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 aliases = { }
|
local aliases = { }
|
||||||
local _list_0 = self:get_stubs(signature)
|
for _index_0 = 1, #signature do
|
||||||
for _index_0 = 1, #_list_0 do
|
local _des_0 = signature[_index_0]
|
||||||
local _des_0 = _list_0[_index_0]
|
|
||||||
local stub, arg_names
|
local stub, arg_names
|
||||||
stub, arg_names = _des_0[1], _des_0[2]
|
stub, arg_names = _des_0[1], _des_0[2]
|
||||||
assert(stub, "NO STUB FOUND: " .. tostring(repr(signature)))
|
assert(stub, "NO STUB FOUND: " .. tostring(repr(signature)))
|
||||||
@ -251,14 +274,18 @@ do
|
|||||||
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,
|
||||||
call = function(self, stub, ...)
|
call = function(self, stub, line_no, ...)
|
||||||
local def = self.defs[stub]
|
local def = self.defs[stub]
|
||||||
|
if def and def.is_macro and self.callstack[#self.callstack] ~= "#macro" then
|
||||||
|
self:error("Attempt to call macro at runtime: " .. tostring(stub) .. "\nThis can be caused by using a macro in a function that is defined before the macro.")
|
||||||
|
end
|
||||||
|
insert(self.callstack, {
|
||||||
|
stub,
|
||||||
|
line_no
|
||||||
|
})
|
||||||
if def == nil then
|
if def == nil then
|
||||||
self:error("Attempt to call undefined function: " .. tostring(stub))
|
self:error("Attempt to call undefined function: " .. tostring(stub))
|
||||||
end
|
end
|
||||||
if def.is_macro and self.callstack[#self.callstack] ~= "#macro" then
|
|
||||||
self:error("Attempt to call macro at runtime: " .. tostring(stub) .. "\nThis can be caused by using a macro in a function that is defined before the macro.")
|
|
||||||
end
|
|
||||||
if not (def.is_macro) then
|
if not (def.is_macro) then
|
||||||
self:assert_permission(stub)
|
self:assert_permission(stub)
|
||||||
end
|
end
|
||||||
@ -276,7 +303,6 @@ 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
|
||||||
insert(self.callstack, stub)
|
|
||||||
local rets = {
|
local rets = {
|
||||||
thunk(self, args)
|
thunk(self, args)
|
||||||
}
|
}
|
||||||
@ -284,13 +310,27 @@ do
|
|||||||
return unpack(rets)
|
return unpack(rets)
|
||||||
end,
|
end,
|
||||||
run_macro = function(self, tree)
|
run_macro = function(self, tree)
|
||||||
local stub, arg_names, args = self:get_stub(tree)
|
local stub = self:get_stub(tree)
|
||||||
|
local args
|
||||||
|
do
|
||||||
|
local _accum_0 = { }
|
||||||
|
local _len_0 = 1
|
||||||
|
local _list_0 = tree.value
|
||||||
|
for _index_0 = 1, #_list_0 do
|
||||||
|
local arg = _list_0[_index_0]
|
||||||
|
if arg.type ~= "Word" then
|
||||||
|
_accum_0[_len_0] = arg
|
||||||
|
_len_0 = _len_0 + 1
|
||||||
|
end
|
||||||
|
end
|
||||||
|
args = _accum_0
|
||||||
|
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(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, unpack(args))
|
local expr, statement = self:call(stub, tree.line_no, unpack(args))
|
||||||
remove(self.callstack)
|
remove(self.callstack)
|
||||||
return expr, statement
|
return expr, statement
|
||||||
end,
|
end,
|
||||||
@ -306,7 +346,7 @@ do
|
|||||||
local _list_0 = self.callstack
|
local _list_0 = self.callstack
|
||||||
for _index_0 = 1, #_list_0 do
|
for _index_0 = 1, #_list_0 do
|
||||||
local caller = _list_0[_index_0]
|
local caller = _list_0[_index_0]
|
||||||
if whiteset[caller] then
|
if whiteset[caller[1]] then
|
||||||
return true
|
return true
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@ -327,7 +367,7 @@ do
|
|||||||
local _list_0 = self.callstack
|
local _list_0 = self.callstack
|
||||||
for _index_0 = 1, #_list_0 do
|
for _index_0 = 1, #_list_0 do
|
||||||
local caller = _list_0[_index_0]
|
local caller = _list_0[_index_0]
|
||||||
if whiteset[caller] then
|
if whiteset[caller[1]] then
|
||||||
return true
|
return true
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@ -338,12 +378,15 @@ do
|
|||||||
self:writeln(tostring(colored.bright("PARSING:")) .. "\n" .. tostring(str))
|
self:writeln(tostring(colored.bright("PARSING:")) .. "\n" .. tostring(str))
|
||||||
end
|
end
|
||||||
str = str:gsub("\r", "")
|
str = str:gsub("\r", "")
|
||||||
|
local old_file = CURRENT_FILE
|
||||||
local old_indent_stack
|
local old_indent_stack
|
||||||
old_indent_stack, indent_stack = indent_stack, {
|
old_indent_stack, indent_stack = indent_stack, {
|
||||||
0
|
0
|
||||||
}
|
}
|
||||||
|
CURRENT_FILE = filename
|
||||||
local tree = nomsu:match(str)
|
local tree = nomsu:match(str)
|
||||||
indent_stack = old_indent_stack
|
indent_stack = old_indent_stack
|
||||||
|
CURRENT_FILE = old_file
|
||||||
assert(tree, "Failed to parse: " .. tostring(str))
|
assert(tree, "Failed to parse: " .. tostring(str))
|
||||||
if self.debug then
|
if self.debug then
|
||||||
self:writeln("PARSE TREE:")
|
self:writeln("PARSE TREE:")
|
||||||
@ -481,7 +524,8 @@ do
|
|||||||
return expr, statement
|
return expr, statement
|
||||||
end
|
end
|
||||||
local args = {
|
local args = {
|
||||||
repr(stub)
|
repr(stub),
|
||||||
|
repr(tree.line_no)
|
||||||
}
|
}
|
||||||
local _list_0 = tree.value
|
local _list_0 = tree.value
|
||||||
for _index_0 = 1, #_list_0 do
|
for _index_0 = 1, #_list_0 do
|
||||||
@ -673,7 +717,7 @@ do
|
|||||||
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, args = { }, { }, { }
|
local stub, arg_names = { }, { }, { }
|
||||||
local _list_0 = x.value
|
local _list_0 = x.value
|
||||||
for _index_0 = 1, #_list_0 do
|
for _index_0 = 1, #_list_0 do
|
||||||
local token = _list_0[_index_0]
|
local token = _list_0[_index_0]
|
||||||
@ -685,14 +729,12 @@ do
|
|||||||
if arg_names then
|
if arg_names then
|
||||||
insert(arg_names, token.value)
|
insert(arg_names, token.value)
|
||||||
end
|
end
|
||||||
insert(args, token)
|
|
||||||
else
|
else
|
||||||
insert(stub, "%")
|
insert(stub, "%")
|
||||||
arg_names = nil
|
arg_names = nil
|
||||||
insert(args, token)
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
return concat(stub, " "), arg_names, args
|
return concat(stub, " "), arg_names
|
||||||
else
|
else
|
||||||
return self:error("Unsupported get stub type: " .. tostring(x.type))
|
return self:error("Unsupported get stub type: " .. tostring(x.type))
|
||||||
end
|
end
|
||||||
@ -754,8 +796,21 @@ do
|
|||||||
self:errorln(colored.bright(colored.yellow(colored.onred(msg))))
|
self:errorln(colored.bright(colored.yellow(colored.onred(msg))))
|
||||||
end
|
end
|
||||||
self:errorln("Callstack:")
|
self:errorln("Callstack:")
|
||||||
|
local maxlen = utils.max((function()
|
||||||
|
local _accum_0 = { }
|
||||||
|
local _len_0 = 1
|
||||||
|
local _list_0 = self.callstack
|
||||||
|
for _index_0 = 1, #_list_0 do
|
||||||
|
local c = _list_0[_index_0]
|
||||||
|
_accum_0[_len_0] = #c[2]
|
||||||
|
_len_0 = _len_0 + 1
|
||||||
|
end
|
||||||
|
return _accum_0
|
||||||
|
end)())
|
||||||
for i = #self.callstack, 1, -1 do
|
for i = #self.callstack, 1, -1 do
|
||||||
self:errorln(" " .. tostring(self.callstack[i]))
|
if self.callstack[i] ~= "#macro" then
|
||||||
|
self:errorln(" " .. tostring(("%-" .. tostring(maxlen) .. "s"):format(self.callstack[i][2])) .. "| " .. tostring(self.callstack[i][1]))
|
||||||
|
end
|
||||||
end
|
end
|
||||||
self:errorln(" <top level>")
|
self:errorln(" <top level>")
|
||||||
self.callstack = { }
|
self.callstack = { }
|
||||||
|
17
nomsu.moon
17
nomsu.moon
@ -210,10 +210,14 @@ class NomsuCompiler
|
|||||||
@write_err("\n")
|
@write_err("\n")
|
||||||
|
|
||||||
def: (signature, thunk, src, is_macro=false)=>
|
def: (signature, thunk, src, is_macro=false)=>
|
||||||
|
if type(signature) == 'string'
|
||||||
|
signature = @get_stubs {signature}
|
||||||
|
elseif type(signature) == 'table' and type(signature[1]) == 'string'
|
||||||
|
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
|
||||||
aliases = {}
|
aliases = {}
|
||||||
for {stub, arg_names} in *@get_stubs(signature)
|
for {stub, arg_names} 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
|
||||||
@ -249,7 +253,8 @@ class NomsuCompiler
|
|||||||
return unpack(rets)
|
return unpack(rets)
|
||||||
|
|
||||||
run_macro: (tree)=>
|
run_macro: (tree)=>
|
||||||
stub,arg_names,args = @get_stub tree
|
stub = @get_stub tree
|
||||||
|
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(stub)} "
|
||||||
@writeln "#{colored.bright "WITH ARGS:"} #{colored.dim repr args}"
|
@writeln "#{colored.bright "WITH ARGS:"} #{colored.dim repr args}"
|
||||||
@ -529,7 +534,7 @@ class NomsuCompiler
|
|||||||
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 args ({msg}) from a single rule def
|
-- Returns a single stub ("say %"), and list of arg names ({"msg"}) 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("'"," '")\gsub("%%%S+","%%")\gsub("%s+"," ")
|
stub = x\gsub("'"," '")\gsub("%%%S+","%%")\gsub("%s+"," ")
|
||||||
@ -538,7 +543,7 @@ class NomsuCompiler
|
|||||||
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"
|
||||||
stub, arg_names, args = {}, {}, {}
|
stub, arg_names = {}, {}, {}
|
||||||
for token in *x.value
|
for token in *x.value
|
||||||
switch token.type
|
switch token.type
|
||||||
when "Word"
|
when "Word"
|
||||||
@ -546,12 +551,10 @@ class NomsuCompiler
|
|||||||
when "Var"
|
when "Var"
|
||||||
insert stub, "%"
|
insert stub, "%"
|
||||||
if arg_names then insert arg_names, token.value
|
if arg_names then insert arg_names, token.value
|
||||||
insert args, token
|
|
||||||
else
|
else
|
||||||
insert stub, "%"
|
insert stub, "%"
|
||||||
arg_names = nil
|
arg_names = nil
|
||||||
insert args, token
|
return concat(stub," "), arg_names
|
||||||
return concat(stub," "), arg_names, args
|
|
||||||
else @error "Unsupported get stub type: #{x.type}"
|
else @error "Unsupported get stub type: #{x.type}"
|
||||||
|
|
||||||
get_stubs: (x)=>
|
get_stubs: (x)=>
|
||||||
|
Loading…
Reference in New Issue
Block a user