2018-11-11 15:50:46 -08:00
|
|
|
#!/usr/bin/env nomsu -V4.10.12.7
|
2018-05-15 18:55:55 -07:00
|
|
|
#
|
2018-01-11 02:07:37 -08:00
|
|
|
This File contains actions for making actions and compile-time actions and some helper
|
|
|
|
functions to make that easier.
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-11-17 14:38:05 -08:00
|
|
|
lua> "NOMSU_CORE_VERSION = 11"
|
|
|
|
lua> "NOMSU_LIB_VERSION = 8"
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
|
|
|
..do
|
2018-09-06 12:46:39 -07:00
|
|
|
local mangle_index = 0
|
|
|
|
function mangler()
|
|
|
|
local my_mangle_index = mangle_index
|
|
|
|
mangle_index = mangle_index + 1
|
|
|
|
return function(varname)
|
2018-09-10 16:36:51 -07:00
|
|
|
return (varname..(("\\3%X"):format(my_mangle_index))):as_lua_id()
|
2018-09-06 12:46:39 -07:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2018-11-09 16:40:36 -08:00
|
|
|
compile.action["define mangler"] = function(compile)
|
|
|
|
return LuaCode("local mangle = mangler()")
|
2018-09-14 19:17:09 -07:00
|
|
|
end"
|
2018-09-10 16:26:08 -07:00
|
|
|
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
2018-11-09 16:40:36 -08:00
|
|
|
..compile.action["1 ->"] = function(compile, \%args, \%body)
|
2018-11-17 14:38:05 -08:00
|
|
|
if \%args and not \%body then \%args, \%body = {}, \%args end
|
2018-11-08 15:23:22 -08:00
|
|
|
local body_lua = SyntaxTree:is_instance(\%body) and compile(\%body) or \%body
|
|
|
|
if SyntaxTree:is_instance(\%body) and \%body.type ~= "Block" then body_lua:prepend("return ") end
|
2018-11-17 14:38:05 -08:00
|
|
|
local lua = LuaCode("(function(")
|
|
|
|
if SyntaxTree:is_instance(\%args) and \%args.type == "Action" then \%args = \%args:get_args()
|
|
|
|
elseif SyntaxTree:is_instance(\%args) and \%args.type == "Var" then \%args = {\%args} end
|
|
|
|
for i, arg in ipairs(\%args) do
|
|
|
|
local arg_lua = SyntaxTree:is_instance(arg) and compile(arg):text() or arg
|
|
|
|
if arg_lua == "..." then
|
|
|
|
if i < #\%args then
|
|
|
|
compile_error_at(SyntaxTree:is_instance(arg) and arg or nil,
|
|
|
|
"Extra arguments must come last.", "Try removing any arguments after (*extra arguments*)")
|
|
|
|
end
|
|
|
|
elseif not arg_lua:is_lua_id() then
|
|
|
|
compile_error_at(SyntaxTree:is_instance(arg) and arg or nil,
|
|
|
|
"This does not compile to a Lua identifier, so it can't be used as a function argument.",
|
|
|
|
"This should probably be a Nomsu variable instead (like %x).")
|
|
|
|
end
|
|
|
|
lua:append(i > 1 and ", " or "", arg_lua)
|
|
|
|
body_lua:remove_free_vars({arg_lua})
|
|
|
|
end
|
2018-06-18 18:10:59 -07:00
|
|
|
body_lua:declare_locals()
|
2018-07-24 13:39:04 -07:00
|
|
|
lua:append(")\\n ", body_lua, "\\nend)")
|
2018-06-18 18:10:59 -07:00
|
|
|
return lua
|
2018-11-17 14:38:05 -08:00
|
|
|
end
|
|
|
|
compile.action["->"] = compile.action["1 ->"]
|
|
|
|
compile.action["for"] = compile.action["1 ->"]"
|
2018-06-18 18:10:59 -07:00
|
|
|
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
2018-11-09 16:40:36 -08:00
|
|
|
..compile.action["what 1 compiles to"] = function(compile, \%action)
|
|
|
|
local lua = LuaCode("compile.action[", \%action.stub:as_lua(), "](")
|
2018-11-17 14:38:05 -08:00
|
|
|
local lua_args = table.map(\%action:get_args(), compile)
|
2018-11-08 15:23:22 -08:00
|
|
|
table.insert(lua_args, 1, "compile")
|
2018-06-18 18:10:59 -07:00
|
|
|
lua:concat_append(lua_args, ", ")
|
|
|
|
lua:append(")")
|
|
|
|
return lua
|
2018-09-14 19:17:09 -07:00
|
|
|
end"
|
2018-06-18 18:10:59 -07:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
2018-06-18 18:10:59 -07:00
|
|
|
|
2018-07-22 15:01:05 -07:00
|
|
|
test:
|
2018-11-08 15:23:22 -08:00
|
|
|
(five) compiles to "5"
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
|
|
|
assume ((five) == 5) or barf "Compile to expression failed."
|
2018-11-08 15:23:22 -08:00
|
|
|
(loc x) compiles to "local x = 99;"
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
|
|
|
lua> "do"
|
|
|
|
loc x
|
|
|
|
assume (%x is 99) or barf "Compile to statements with locals failed."
|
|
|
|
lua> "end"
|
|
|
|
assume (%x is (nil)) or barf "Failed to properly localize a variable."
|
2018-10-30 23:42:04 -07:00
|
|
|
(asdf) compiles to:
|
2018-07-22 13:59:08 -07:00
|
|
|
%tmp = ""
|
|
|
|
return (Lua %tmp)
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
|
|
|
asdf
|
|
|
|
assume (%tmp is (nil)) or barf "compile to is leaking variables"
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
2018-11-09 16:40:36 -08:00
|
|
|
..compile.action["1 compiles to"] = function(compile, \%action, \%body)
|
|
|
|
local \%args = List{\(\%compile), unpack(\%action:get_args())}
|
2018-11-08 15:23:22 -08:00
|
|
|
if \%body.type == "Text" then
|
|
|
|
\%body = SyntaxTree{source=\%body.source, type="Action", "Lua", \%body}
|
|
|
|
end
|
2018-11-09 16:40:36 -08:00
|
|
|
return LuaCode("compile.action[", \%action.stub:as_lua(),
|
2018-10-30 23:42:04 -07:00
|
|
|
"] = ", \(what (%args -> %body) compiles to))
|
2018-11-06 15:13:55 -08:00
|
|
|
end"
|
|
|
|
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
(%actions all compile to %body) compiles to:
|
|
|
|
lua> "\
|
|
|
|
..if \%actions.type ~= "List" then
|
2018-11-08 15:23:22 -08:00
|
|
|
compile_error(\%actions, "This should be a list of actions.")
|
2018-11-06 15:13:55 -08:00
|
|
|
end
|
2018-11-09 16:40:36 -08:00
|
|
|
local lua = \(what (%actions.1 compiles to %body) compiles to)
|
|
|
|
local \%args = List{\(\%compile), unpack(\%actions[1]:get_args())}
|
2018-06-18 18:10:59 -07:00
|
|
|
for i=2,#\%actions do
|
|
|
|
local alias = \%actions[i]
|
2018-11-09 16:40:36 -08:00
|
|
|
local \%alias_args = List{\(\%compile), unpack(alias:get_args())}
|
2018-11-08 15:23:22 -08:00
|
|
|
lua:append("\\ncompile.action[", alias.stub:as_lua(), "] = ")
|
2018-11-06 15:13:55 -08:00
|
|
|
if \%alias_args == \%args then
|
2018-11-08 15:23:22 -08:00
|
|
|
lua:append("compile.action[", \%actions[1].stub:as_lua(), "]")
|
2018-06-18 15:44:29 -07:00
|
|
|
else
|
2018-11-06 15:13:55 -08:00
|
|
|
lua:append(\(what (%alias_args -> \(what %actions.1 compiles to)) compiles to))
|
2018-06-14 21:59:25 -07:00
|
|
|
end
|
2018-06-18 15:44:29 -07:00
|
|
|
end
|
2018-11-06 15:13:55 -08:00
|
|
|
return lua"
|
2018-06-18 18:10:59 -07:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
2018-06-18 18:10:59 -07:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
2018-11-08 15:23:22 -08:00
|
|
|
(foo %x) means "outer"
|
2018-11-19 17:21:08 -08:00
|
|
|
with {((foo %)'s meaning)}:
|
2018-10-30 23:42:04 -07:00
|
|
|
(foo %x) means:
|
2018-07-22 13:59:08 -07:00
|
|
|
%y = (%x + 1)
|
|
|
|
return %y
|
|
|
|
assume ((foo 10) == 11) or barf "Action didn't work."
|
|
|
|
assume (%y is (nil)) or barf "Action leaked a local into globals."
|
2018-10-30 23:42:04 -07:00
|
|
|
(baz %) parses as (foo %)
|
2018-07-22 13:59:08 -07:00
|
|
|
assume ((foo 1) == "outer")
|
2018-11-06 15:13:55 -08:00
|
|
|
|
|
|
|
(%action means %body) compiles to:
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
2018-11-17 14:38:05 -08:00
|
|
|
..
|
|
|
|
local lua = LuaCode()
|
|
|
|
local fn_name = \%action.stub:as_lua_id()
|
|
|
|
if \%action.target then lua:append(compile(\%action.target), ".")
|
|
|
|
else lua:add_free_vars({fn_name}) end
|
|
|
|
lua:append(fn_name, " = ", \(what (%action -> %body) compiles to), ";")
|
2018-11-06 15:13:55 -08:00
|
|
|
return lua"
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-11-06 15:13:55 -08:00
|
|
|
(%actions all mean %body) compiles to:
|
|
|
|
lua> "\
|
|
|
|
..local fn_name = \%actions[1].stub:as_lua_id()
|
2018-11-17 14:38:05 -08:00
|
|
|
local target = \%actions[1].target and compile(\%actions[1].target) or nil
|
2018-11-06 15:13:55 -08:00
|
|
|
local \%args = List(\%actions[1]:get_args())
|
2018-11-09 16:40:36 -08:00
|
|
|
local lua = \(what (%actions.1 means %body) compiles to)
|
2018-06-18 18:10:59 -07:00
|
|
|
for i=2,#\%actions do
|
|
|
|
local alias = \%actions[i]
|
2018-09-10 16:36:51 -07:00
|
|
|
local alias_name = alias.stub:as_lua_id()
|
2018-11-06 15:13:55 -08:00
|
|
|
local \%alias_args = List(alias:get_args())
|
2018-11-17 14:38:05 -08:00
|
|
|
lua:append("\\n")
|
|
|
|
if alias.target then
|
|
|
|
lua:append(compile(alias.target), ".")
|
|
|
|
else
|
|
|
|
lua:add_free_vars({alias_name})
|
|
|
|
end
|
|
|
|
lua:append(alias_name, " = ")
|
2018-11-06 15:13:55 -08:00
|
|
|
if \%args == \%alias_args then
|
2018-11-17 14:38:05 -08:00
|
|
|
if target then lua:append(target, ".") end
|
|
|
|
lua:append(fn_name, ";")
|
2018-06-18 18:10:59 -07:00
|
|
|
else
|
2018-11-17 14:38:05 -08:00
|
|
|
lua:append(\(what (%alias_args -> %actions.1) compiles to), ";")
|
2018-06-18 18:10:59 -07:00
|
|
|
end
|
2018-06-18 15:44:29 -07:00
|
|
|
end
|
2018-09-14 19:17:09 -07:00
|
|
|
return lua"
|
2018-06-18 15:44:29 -07:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
2018-11-11 15:50:46 -08:00
|
|
|
externally (baz1) means:
|
|
|
|
return "baz1"
|
2018-10-30 23:42:04 -07:00
|
|
|
externally (baz2) means "baz2"
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
|
|
|
assume ((baz1) == "baz1")
|
|
|
|
assume ((baz2) == "baz2")
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-11-06 15:13:55 -08:00
|
|
|
(externally %action means %body) compiles to:
|
|
|
|
lua> "\
|
|
|
|
..local lua = \(what (%action means %body) compiles to)
|
|
|
|
lua:remove_free_vars({\%action.stub:as_lua_id()})
|
|
|
|
return lua"
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-11-06 15:13:55 -08:00
|
|
|
(externally %actions all mean %body) compiles to:
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
2018-11-06 15:13:55 -08:00
|
|
|
..local lua = \(what (%actions all mean %body) compiles to)
|
2018-09-10 16:36:51 -07:00
|
|
|
lua:remove_free_vars(table.map(\%actions, function(a) return a.stub:as_lua_id() end))
|
2018-09-14 19:17:09 -07:00
|
|
|
return lua"
|
2018-06-18 18:10:59 -07:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
2018-11-02 14:38:24 -07:00
|
|
|
assume (((say %)'s meaning) == (=lua "say"))
|
2018-07-30 15:05:41 -07:00
|
|
|
|
2018-11-11 15:50:46 -08:00
|
|
|
(%action's meaning) compiles to (Lua (%action.stub as lua id))
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
2018-10-30 23:42:04 -07:00
|
|
|
(swap %x and %y) parses as (..)
|
2018-07-22 13:59:08 -07:00
|
|
|
do:
|
|
|
|
%tmp = %x
|
|
|
|
%x = %y
|
|
|
|
%y = %tmp
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
2018-11-17 14:38:05 -08:00
|
|
|
[%1, %2] = [1, 2]
|
2018-07-22 13:59:08 -07:00
|
|
|
swap %1 and %2
|
2018-09-14 19:17:09 -07:00
|
|
|
assume ((%1 == 2) and (%2 == 1)) or barf "\
|
|
|
|
..'parse % as %' failed on 'swap % and %'"
|
2018-11-17 14:38:05 -08:00
|
|
|
[%tmp, %tmp2] = [1, 2]
|
2018-07-22 13:59:08 -07:00
|
|
|
swap %tmp and %tmp2
|
2018-09-14 19:17:09 -07:00
|
|
|
assume ((%tmp == 2) and (%tmp2 == 1)) or barf "\
|
|
|
|
..'parse % as %' variable mangling failed."
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-11-06 15:13:55 -08:00
|
|
|
(%actions all parse as %body) compiles to:
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
|
|
|
..local replacements = {}
|
2018-11-06 15:13:55 -08:00
|
|
|
if \%actions.type ~= "List" then
|
2018-11-08 15:23:22 -08:00
|
|
|
compile_error(\%actions, "This should be a list.")
|
2018-11-06 15:13:55 -08:00
|
|
|
end
|
2018-06-18 15:44:29 -07:00
|
|
|
for i,arg in ipairs(\%actions[1]:get_args()) do
|
2018-11-08 15:23:22 -08:00
|
|
|
replacements[arg[1]] = compile(arg):text()
|
2018-06-18 15:44:29 -07:00
|
|
|
end
|
|
|
|
local function make_tree(t)
|
2018-10-31 15:54:18 -07:00
|
|
|
if SyntaxTree:is_instance(t) and t.type == "Var" then
|
2018-08-28 15:08:00 -07:00
|
|
|
if replacements[t[1]] then
|
|
|
|
return replacements[t[1]]
|
|
|
|
else
|
2018-11-11 15:50:46 -08:00
|
|
|
return "SyntaxTree{mangle("..t[1]:as_lua().."), type="..t.type:as_lua()..", source="..tostring(\
|
|
|
|
..t.source):as_lua().."}"
|
2018-08-28 15:08:00 -07:00
|
|
|
end
|
2018-10-31 15:54:18 -07:00
|
|
|
elseif SyntaxTree:is_instance(t) then
|
2018-08-28 15:08:00 -07:00
|
|
|
local ret = {}
|
|
|
|
local i = 1
|
|
|
|
for k, v in pairs(t) do
|
|
|
|
if k == i then
|
|
|
|
ret[#ret+1] = make_tree(t[i])
|
|
|
|
i = i + 1
|
|
|
|
elseif k == "source" then
|
2018-09-18 19:48:58 -07:00
|
|
|
ret[#ret+1] = k.."= "..tostring(v):as_lua()
|
2018-11-02 14:38:24 -07:00
|
|
|
elseif lua_type_of(k) == 'string' and k:match("[_a-zA-Z][_a-zA-Z0-9]*") then
|
2018-08-28 15:08:00 -07:00
|
|
|
ret[#ret+1] = k.."= "..make_tree(v)
|
|
|
|
else
|
|
|
|
ret[#ret+1] = "["..make_tree(k).."]= "..make_tree(v)
|
|
|
|
end
|
|
|
|
end
|
2018-10-31 15:54:18 -07:00
|
|
|
return "SyntaxTree{"..table.concat(ret, ", ").."}"
|
2018-11-02 14:38:24 -07:00
|
|
|
elseif lua_type_of(t) == 'number' then
|
2018-09-18 19:48:58 -07:00
|
|
|
return tostring(t)
|
2018-06-18 15:44:29 -07:00
|
|
|
else
|
2018-09-18 19:48:58 -07:00
|
|
|
return t:as_lua()
|
2018-06-12 20:06:33 -07:00
|
|
|
end
|
2018-06-18 15:44:29 -07:00
|
|
|
end
|
2018-11-09 16:40:36 -08:00
|
|
|
local \%new_body = LuaCode:from(\%body.source,
|
2018-09-06 12:46:39 -07:00
|
|
|
"local mangle = mangler()",
|
2018-09-17 15:29:48 -07:00
|
|
|
"\\nreturn ", make_tree(\%body))
|
2018-11-06 15:13:55 -08:00
|
|
|
local ret = \(what (%actions all compile to %new_body) compiles to)
|
2018-09-14 19:17:09 -07:00
|
|
|
return ret"
|
2018-06-18 15:44:29 -07:00
|
|
|
|
2018-11-11 15:50:46 -08:00
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
2018-11-06 15:13:55 -08:00
|
|
|
|
|
|
|
[%action parses as %body] all parse as ([%action] all parse as %body)
|
2018-11-17 14:38:05 -08:00
|
|
|
#(%tree as lua expr) compiles to "compile(\(=lua "compile(\%tree, true)"), true)"
|
|
|
|
externally (%tree as lua expr) means:
|
|
|
|
lua> "\
|
|
|
|
..local tree_lua = compile(\%tree)
|
|
|
|
if \%tree.type == 'Block' then
|
|
|
|
tree_lua = LuaCode:from(\%tree.source, '(function()\n ', tree_lua, '\nend)()')
|
|
|
|
end
|
|
|
|
return tree_lua"
|
2017-12-11 17:53:23 -08:00
|
|
|
|
2018-05-16 18:12:56 -07:00
|
|
|
|
2018-10-30 23:42:04 -07:00
|
|
|
externally [%var as lua identifier, %var as lua id] all mean:
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
2018-11-02 14:38:24 -07:00
|
|
|
..if lua_type_of(\%var) == 'string' then return \%var:as_lua_id()
|
2018-10-31 15:54:18 -07:00
|
|
|
elseif SyntaxTree:is_instance(\%var, 'Var') then return \%var[1]:as_lua_id()
|
|
|
|
elseif SyntaxTree:is_instance(\%var) then
|
2018-09-21 00:30:28 -07:00
|
|
|
local lua = \(%var as lua expr)
|
2018-11-02 14:38:24 -07:00
|
|
|
if not lua:text():match("^[_a-zA-Z][_a-zA-Z0-9]*$") then
|
2018-11-08 15:23:22 -08:00
|
|
|
compile_error(\%var, "This is not a valid Lua identifier.")
|
2018-09-21 00:30:28 -07:00
|
|
|
end
|
|
|
|
return lua
|
2018-09-06 12:46:39 -07:00
|
|
|
else error("Unknown type: "..tostring(\%var))
|
2018-09-14 19:17:09 -07:00
|
|
|
end"
|
2018-06-18 15:44:29 -07:00
|
|
|
|
2018-11-17 14:38:05 -08:00
|
|
|
test:
|
|
|
|
(num args (*extra arguments*)) means (select "#" (*extra arguments*))
|
|
|
|
assume (num args 1 2 3) == 3
|
|
|
|
(extra args (*extra arguments*)) means [*extra arguments*]
|
|
|
|
assume (extra args 1 2 3) == [1, 2, 3]
|
|
|
|
(third arg (*extra arguments*)) means (select 3 (*extra arguments*))
|
|
|
|
assume (third arg 5 6 7 8) == 7
|
|
|
|
(*extra arguments*) compiles to "..."
|
2018-06-26 15:52:38 -07:00
|
|
|
|
2018-11-08 15:23:22 -08:00
|
|
|
(% is syntax tree) compiles to "SyntaxTree:is_instance(\(% as lua expr))"
|
|
|
|
externally (% is %kind syntax tree) means (..)
|
|
|
|
=lua "SyntaxTree:is_instance(\%) and \%.type == \%kind"
|
2018-07-14 14:52:28 -07:00
|
|
|
|
2018-11-08 15:23:22 -08:00
|
|
|
(%tree with %t -> %replacement) compiles to "\
|
|
|
|
..\(%tree as lua expr):map(function(\(%t as lua expr))
|
2018-11-11 15:50:46 -08:00
|
|
|
\(..)
|
|
|
|
(%replacement as lua) if (%replacement.type == "Block") else "\
|
|
|
|
..return \(%replacement as lua expr)"
|
|
|
|
..
|
2018-11-08 15:23:22 -08:00
|
|
|
end)"
|
2018-01-25 17:34:49 -08:00
|
|
|
|
2018-10-30 23:42:04 -07:00
|
|
|
externally (%tree with vars %replacements) means (..)
|
2018-09-14 19:17:09 -07:00
|
|
|
=lua "\
|
|
|
|
..\%tree:map(function(\%t)
|
2018-07-17 14:12:11 -07:00
|
|
|
if \%t.type == "Var" then
|
|
|
|
return \%replacements[\%t[1]]
|
2018-06-23 00:57:31 -07:00
|
|
|
end
|
2018-09-14 19:17:09 -07:00
|
|
|
end)"
|
2018-06-23 00:57:31 -07:00
|
|
|
|
2018-11-08 15:23:22 -08:00
|
|
|
(tree %tree with vars %replacements) compiles to "\
|
|
|
|
..\(=lua "(\%tree):as_lua()"):map(function(t)
|
|
|
|
if t.type == "Var" then
|
|
|
|
return \(%replacements as lua expr)[t[1]]
|
|
|
|
end
|
|
|
|
end)"
|
2018-08-27 13:38:58 -07:00
|
|
|
|
2018-11-08 15:23:22 -08:00
|
|
|
(%tree has subtree %match_tree) compiles to "\
|
|
|
|
..(function()
|
|
|
|
local match_tree = \(%match_tree as lua expr)
|
2018-11-17 14:38:05 -08:00
|
|
|
for subtree in coroutine_wrap(function() \(%tree as lua expr):map(yield) end) do
|
2018-11-08 15:23:22 -08:00
|
|
|
if subtree == match_tree then return true end
|
|
|
|
end
|
|
|
|
end)()"
|
2018-07-22 13:59:08 -07:00
|
|
|
|
2018-10-30 23:42:04 -07:00
|
|
|
externally (match %tree with %patt) means:
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
2018-11-06 15:13:55 -08:00
|
|
|
..if \%patt.type == "Var" then return Dict{[\%patt[1]]=\%tree} end
|
2018-08-27 13:38:58 -07:00
|
|
|
if \%patt.type == "Action" and \%patt.stub ~= \%tree.stub then return nil end
|
|
|
|
if #\%patt ~= #\%tree then return nil end
|
2018-11-06 15:13:55 -08:00
|
|
|
local matches = Dict{}
|
2018-08-27 13:38:58 -07:00
|
|
|
for \%i=1,#\%patt do
|
2018-10-31 15:54:18 -07:00
|
|
|
if SyntaxTree:is_instance(\%tree[\%i]) then
|
2018-08-27 13:38:58 -07:00
|
|
|
local submatch = \(match %tree.%i with %patt.%i)
|
|
|
|
if not submatch then return nil end
|
|
|
|
for k,v in pairs(submatch) do
|
|
|
|
if matches[k] and matches[k] ~= v then return nil end
|
|
|
|
matches[k] = v
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2018-09-14 19:17:09 -07:00
|
|
|
return matches"
|
2018-08-27 13:38:58 -07:00
|
|
|
|
2018-10-30 23:42:04 -07:00
|
|
|
externally (%tree with %patt ~> %replacement) means:
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
|
|
|
..return \%tree:map(function(\%t)
|
2018-08-27 13:38:58 -07:00
|
|
|
local \%vars = \(match %t with %patt)
|
|
|
|
if not \%vars then return nil end
|
|
|
|
for \%k,\%v in pairs(\%vars) do
|
|
|
|
\%vars[\%k] = \(%v with %patt ~> %replacement)
|
|
|
|
end
|
|
|
|
return \%replacement:map(function(\%t)
|
|
|
|
if \%t.type == "Var" then
|
|
|
|
return \%vars[\%t[1]]
|
|
|
|
end
|
|
|
|
end)
|
2018-09-14 19:17:09 -07:00
|
|
|
end)"
|
2018-08-27 13:38:58 -07:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
2018-11-11 15:50:46 -08:00
|
|
|
assume ((quote "one\n\"two\"") == "\"one\\n\\\"two\\\"\"")
|
2018-07-30 15:05:41 -07:00
|
|
|
|
2018-11-11 15:50:46 -08:00
|
|
|
(quote %s) compiles to "tostring(\(%s as lua expr)):as_lua()"
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
2018-09-26 13:05:28 -07:00
|
|
|
assume (lua type of {}) == "table"
|
2018-09-21 00:30:28 -07:00
|
|
|
assume (type of {}) == "Dict"
|
|
|
|
assume ({} is a "Dict")
|
|
|
|
assume ("" is text)
|
|
|
|
assume ("" isn't a "Dict")
|
2018-10-30 23:42:04 -07:00
|
|
|
externally (% is text) means (=lua "\(lua type of %) == 'string'")
|
|
|
|
externally [% is not text, % isn't text] all mean (..)
|
|
|
|
=lua "\(lua type of %) ~= 'string'"
|
2018-09-21 00:30:28 -07:00
|
|
|
|
2018-10-30 23:42:04 -07:00
|
|
|
externally (type of %) means:
|
2018-09-21 00:30:28 -07:00
|
|
|
lua> "\
|
2018-10-30 23:42:04 -07:00
|
|
|
..local lua_type = \(lua type of %)
|
2018-09-21 00:30:28 -07:00
|
|
|
if lua_type == 'string' then return 'Text'
|
2018-11-06 15:13:55 -08:00
|
|
|
elseif lua_type == 'table' or lua_type == 'userdata' then
|
2018-09-21 00:30:28 -07:00
|
|
|
local mt = getmetatable(\%)
|
|
|
|
if mt and mt.__type then return mt.__type end
|
2018-11-06 15:13:55 -08:00
|
|
|
end
|
|
|
|
return lua_type"
|
2018-10-30 23:42:04 -07:00
|
|
|
|
|
|
|
[% is a %type, % is an %type] all parse as ((type of %) == %type)
|
2018-11-11 15:50:46 -08:00
|
|
|
[% isn't a %type, % isn't an %type, % is not a %type, % is not an %type] \
|
|
|
|
..all parse as ((type of %) != %type)
|
2018-07-22 13:59:08 -07:00
|
|
|
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
test:
|
|
|
|
assume ((run "return (2 + 99)") == 101)
|
|
|
|
external %passed = (no)
|
2018-09-15 20:20:40 -07:00
|
|
|
run "external %passed = (yes)"
|
2018-07-22 13:59:08 -07:00
|
|
|
assume %passed
|
2018-11-08 15:23:22 -08:00
|
|
|
assume (run \(return \(\(5) + \(5)))) == 10
|
2018-11-11 15:50:46 -08:00
|
|
|
(run %nomsu_code) compiles to "run_1_in(\(%nomsu_code as lua expr), _ENV)"
|
2018-11-08 15:23:22 -08:00
|
|
|
[compile %block, compiled %block, %block compiled] all compile to "\
|
|
|
|
..compile(\(%block as lua))"
|
2018-01-25 17:34:49 -08:00
|
|
|
|
2018-11-17 14:38:05 -08:00
|
|
|
test:
|
|
|
|
(foo) means: return 100 200 300
|
|
|
|
assume (select 2 (foo)) == 200
|
2018-06-18 15:44:29 -07:00
|
|
|
# Return statement is wrapped in a do..end block because Lua is unhappy if you
|
|
|
|
put code after a return statement, unless you wrap it in a block.
|
2018-11-17 14:38:05 -08:00
|
|
|
(return (*extra arguments*)) compiles to:
|
|
|
|
lua> "\
|
|
|
|
..local lua = \(Lua "do return ")
|
|
|
|
for i=1,select('#',...) do
|
|
|
|
if i > 1 then lua:append(", ") end
|
|
|
|
lua:append(_1_as_lua((select(i, ...))))
|
|
|
|
end
|
|
|
|
lua:append(" end")
|
|
|
|
return lua"
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-01-11 18:51:21 -08:00
|
|
|
# Literals
|
2018-11-08 15:23:22 -08:00
|
|
|
(yes) compiles to "true"
|
|
|
|
(no) compiles to "false"
|
|
|
|
[nothing, nil, null] all compile to "nil"
|
|
|
|
(Nomsu syntax version) compiles to "NOMSU_SYNTAX_VERSION"
|
|
|
|
(Nomsu compiler version) compiles to "NOMSU_COMPILER_VERSION"
|
|
|
|
(core version) compiles to "NOMSU_CORE_VERSION"
|
|
|
|
(lib version) compiles to "NOMSU_LIB_VERSION"
|
|
|
|
(command line args) compiles to "command_line_args"
|
2018-07-17 23:08:13 -07:00
|
|
|
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
2018-11-08 15:23:22 -08:00
|
|
|
(with local compile actions %body) compiles to "\
|
|
|
|
..do
|
|
|
|
--local compile = _1_forked(compile)
|
|
|
|
local old_action = compile.action
|
|
|
|
compile.action = _1_forked(old_action)
|
2018-11-09 14:48:23 -08:00
|
|
|
\(%body as lua)
|
2018-11-08 15:23:22 -08:00
|
|
|
compile.action = old_action
|
|
|
|
end"
|
2018-06-26 15:52:38 -07:00
|
|
|
|
2018-10-30 23:42:04 -07:00
|
|
|
externally (Nomsu version) means:
|
2018-11-11 15:50:46 -08:00
|
|
|
return "\(Nomsu syntax version).\(core version).\(Nomsu compiler version).\(lib version)"
|