2018-09-14 19:17:09 -07:00
|
|
|
#!/usr/bin/env nomsu -V4.8.8.6
|
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.
|
2017-09-21 00:10:26 -07:00
|
|
|
|
2018-09-10 16:26:08 -07:00
|
|
|
lua> "NOMSU_CORE_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
|
|
|
|
COMPILE_ACTIONS["define mangler"] = function(nomsu, tree)
|
|
|
|
return LuaCode(tree.source, "local mangle_1 = 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> "\
|
|
|
|
..COMPILE_ACTIONS["1 -> 2"] = function(nomsu, tree, \%args, \%body)
|
2018-07-24 13:39:04 -07:00
|
|
|
local lua = LuaCode.Value(tree.source, "(function(")
|
2018-06-18 18:10:59 -07:00
|
|
|
if AST.is_syntax_tree(\%args, "Action") then \%args = \%args:get_args() end
|
2018-08-28 15:08:00 -07:00
|
|
|
local lua_args = table.map(\%args, function(a) return AST.is_syntax_tree(a) and tostring(nomsu:compile(a)) or a end)
|
2018-06-18 18:10:59 -07:00
|
|
|
lua:concat_append(lua_args, ", ")
|
2018-08-28 15:08:00 -07:00
|
|
|
local body_lua = AST.is_syntax_tree(\%body) and nomsu:compile(\%body):as_statements("return ") or \%body
|
2018-06-18 18:10:59 -07:00
|
|
|
body_lua:remove_free_vars(lua_args)
|
|
|
|
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-09-14 19:17:09 -07:00
|
|
|
end"
|
2018-06-18 18:10:59 -07:00
|
|
|
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
|
|
|
..COMPILE_ACTIONS["compile as 1"] = function(nomsu, tree, \%action)
|
2018-08-27 13:38:58 -07:00
|
|
|
local lua = LuaCode.Value(tree.source, "COMPILE_ACTIONS[", repr(\%action.stub), "](")
|
2018-08-28 15:08:00 -07:00
|
|
|
local lua_args = table.map(\%action:get_args(), function(a) return nomsu:compile(a) end)
|
2018-06-18 18:10:59 -07:00
|
|
|
table.insert(lua_args, 1, "nomsu")
|
|
|
|
table.insert(lua_args, 2, "tree")
|
|
|
|
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:
|
|
|
|
compile [five] to (Lua value "5")
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
|
|
|
assume ((five) == 5) or barf "Compile to expression failed."
|
2018-09-06 12:46:39 -07:00
|
|
|
compile [loc x] to (Lua "local x = 99;")
|
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."
|
|
|
|
compile [asdf] to:
|
|
|
|
%tmp = ""
|
|
|
|
return (Lua %tmp)
|
|
|
|
test:
|
|
|
|
asdf
|
|
|
|
assume (%tmp is (nil)) or barf "compile to is leaking variables"
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
|
|
|
..COMPILE_ACTIONS["compile 1 to 2"] = function(nomsu, tree, \%actions, \%body)
|
2018-08-28 15:08:00 -07:00
|
|
|
local \%args = {"nomsu", "tree", unpack(table.map(\%actions[1]:get_args(), function(a) return tostring(nomsu:compile(\
|
2018-07-20 20:27:15 -07:00
|
|
|
..a)) end))}
|
2018-08-27 13:38:58 -07:00
|
|
|
local lua = LuaCode(tree.source, "COMPILE_ACTIONS[", repr(\%actions[1].stub),
|
2018-07-17 23:08:13 -07:00
|
|
|
"] = ", \(compile as (%args -> %body)))
|
2018-06-18 18:10:59 -07:00
|
|
|
for i=2,#\%actions do
|
|
|
|
local alias = \%actions[i]
|
2018-08-28 15:08:00 -07:00
|
|
|
local \%alias_args = {"nomsu", "tree", unpack(table.map(alias:get_args(), function(a) return tostring(nomsu:compile(\
|
2018-07-20 20:27:15 -07:00
|
|
|
..a)) end))}
|
2018-08-27 13:38:58 -07:00
|
|
|
lua:append("\\nCOMPILE_ACTIONS[", repr(alias.stub), "] = ")
|
2018-06-18 18:10:59 -07:00
|
|
|
if utils.equivalent(\%args, \%alias_args) then
|
2018-08-27 13:38:58 -07:00
|
|
|
lua:append("COMPILE_ACTIONS[", repr(\%actions[1].stub), "]")
|
2018-06-18 15:44:29 -07:00
|
|
|
else
|
2018-06-18 18:10:59 -07:00
|
|
|
lua:append("function(")
|
|
|
|
lua:concat_append(\%alias_args, ", ")
|
2018-08-27 13:38:58 -07:00
|
|
|
lua:append(")\\n return COMPILE_ACTIONS[", repr(\%actions[1].stub), "](")
|
2018-06-18 18:10:59 -07:00
|
|
|
lua:concat_append(\%args, ", ")
|
2018-07-17 23:08:13 -07:00
|
|
|
lua:append(")\\nend")
|
2018-06-14 21:59:25 -07:00
|
|
|
end
|
2018-06-18 15:44:29 -07:00
|
|
|
end
|
2018-06-18 18:10:59 -07:00
|
|
|
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-08-28 15:08:00 -07:00
|
|
|
compile [call %fn with %args] to:
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
|
|
|
..local lua = LuaCode.Value(tree.source, nomsu:compile(\%fn), "(")
|
2018-08-28 15:08:00 -07:00
|
|
|
lua:concat_append(table.map(\%args, function(a) return nomsu:compile(a) end), ", ")
|
2018-06-23 00:57:31 -07:00
|
|
|
lua:append(")")
|
2018-09-14 19:17:09 -07:00
|
|
|
return lua"
|
2018-06-23 00:57:31 -07:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
2018-07-22 15:01:05 -07:00
|
|
|
local action [foo %x]: return "outer"
|
2018-07-22 13:59:08 -07:00
|
|
|
with local [action (foo %)]:
|
|
|
|
local action [foo %x]:
|
|
|
|
%y = (%x + 1)
|
|
|
|
return %y
|
2018-07-22 15:01:05 -07:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
assume ((foo 10) == 11) or barf "Action didn't work."
|
|
|
|
assume (%y is (nil)) or barf "Action leaked a local into globals."
|
|
|
|
parse [baz %] as (foo %)
|
2018-07-22 15:01:05 -07:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
assume ((foo 1) == "outer")
|
2018-08-28 15:08:00 -07:00
|
|
|
compile [local action %actions %body] to:
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
|
|
|
..local fn_name = \%actions[1].stub:as_lua_id()
|
2018-08-28 15:08:00 -07:00
|
|
|
local \%args = table.map(\%actions[1]:get_args(), function(a) return tostring(nomsu:compile(a)) end)
|
2018-07-17 23:08:13 -07:00
|
|
|
local lua = LuaCode(tree.source, fn_name, " = ", \(compile as (%args -> %body)))
|
2018-06-18 18:10:59 -07:00
|
|
|
lua:add_free_vars({fn_name})
|
|
|
|
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-06-18 18:10:59 -07:00
|
|
|
lua:add_free_vars({alias_name})
|
2018-08-28 15:08:00 -07:00
|
|
|
local \%alias_args = table.map(alias:get_args(), function(a) return tostring(nomsu:compile(a)) end)
|
2018-07-17 23:08:13 -07:00
|
|
|
lua:append("\\n", alias_name, " = ")
|
2018-06-18 18:10:59 -07:00
|
|
|
if utils.equivalent(\%args, \%alias_args) then
|
|
|
|
lua:append(fn_name)
|
|
|
|
else
|
|
|
|
lua:append("function(")
|
|
|
|
lua:concat_append(\%alias_args, ", ")
|
2018-07-17 23:08:13 -07:00
|
|
|
lua:append(")\\n return ", fn_name, "(")
|
2018-06-18 18:10:59 -07:00
|
|
|
lua:concat_append(\%args, ", ")
|
2018-07-17 23:08:13 -07:00
|
|
|
lua:append(")\\nend")
|
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:
|
|
|
|
action [baz1]: return "baz1"
|
|
|
|
action [baz2] "baz2"
|
|
|
|
test:
|
|
|
|
assume ((baz1) == "baz1")
|
|
|
|
assume ((baz2) == "baz2")
|
2018-07-17 23:08:13 -07:00
|
|
|
compile [action %actions %body] to (..)
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
|
|
|
..local lua = \(compile as (local action %actions %body))
|
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-08-31 15:21:47 -07:00
|
|
|
assume ((action (say %)) == (=lua "say_1"))
|
2018-07-22 13:59:08 -07:00
|
|
|
compile [action %action] to (Lua value (%action as lua id))
|
2018-07-30 15:05:41 -07:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
|
|
|
parse [swap %x and %y] as (..)
|
|
|
|
do:
|
|
|
|
%tmp = %x
|
|
|
|
%x = %y
|
|
|
|
%y = %tmp
|
|
|
|
test:
|
|
|
|
set {%1:1, %2:2}
|
|
|
|
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-07-22 13:59:08 -07:00
|
|
|
set {%tmp:1, %tmp2:2}
|
|
|
|
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-07-17 23:08:13 -07:00
|
|
|
compile [parse %actions as %body] to (..)
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
|
|
|
..local replacements = {}
|
2018-06-18 15:44:29 -07:00
|
|
|
for i,arg in ipairs(\%actions[1]:get_args()) do
|
2018-08-28 15:08:00 -07:00
|
|
|
replacements[arg[1]] = tostring(nomsu:compile(arg))
|
2018-06-18 15:44:29 -07:00
|
|
|
end
|
|
|
|
local function make_tree(t)
|
2018-08-28 15:08:00 -07:00
|
|
|
if AST.is_syntax_tree(t, "Var") then
|
|
|
|
if replacements[t[1]] then
|
|
|
|
return replacements[t[1]]
|
|
|
|
else
|
2018-09-06 12:46:39 -07:00
|
|
|
return t.type.."{mangle("..repr(t[1]).."), source="..repr(tostring(t.source)).."}"
|
2018-08-28 15:08:00 -07:00
|
|
|
end
|
|
|
|
elseif AST.is_syntax_tree(t) then
|
|
|
|
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
|
|
|
|
ret[#ret+1] = k.."= "..repr(tostring(v))
|
|
|
|
elseif type(k) == 'string' and k:match("[_a-zA-Z][_a-zA-Z0-9]*") then
|
|
|
|
ret[#ret+1] = k.."= "..make_tree(v)
|
|
|
|
else
|
|
|
|
ret[#ret+1] = "["..make_tree(k).."]= "..make_tree(v)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return t.type.."{"..table.concat(ret, ", ").."}"
|
2018-06-18 15:44:29 -07:00
|
|
|
else
|
2018-08-28 15:08:00 -07:00
|
|
|
return repr(t)
|
2018-06-12 20:06:33 -07:00
|
|
|
end
|
2018-06-18 15:44:29 -07:00
|
|
|
end
|
2018-06-18 18:10:59 -07:00
|
|
|
local \%new_body = LuaCode(\%body.source,
|
2018-09-06 12:46:39 -07:00
|
|
|
"local mangle = mangler()",
|
2018-07-17 23:08:13 -07:00
|
|
|
"\\nlocal tree = ", make_tree(\%body),
|
2018-08-28 15:08:00 -07:00
|
|
|
"\\nlocal lua = nomsu:compile(tree)",
|
|
|
|
"\\nreturn lua")
|
2018-07-17 23:08:13 -07:00
|
|
|
local ret = \(compile as (compile %actions to %new_body))
|
2018-09-14 19:17:09 -07:00
|
|
|
return ret"
|
2018-06-18 15:44:29 -07:00
|
|
|
|
2018-07-18 17:55:29 -07:00
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
action [%tree as lua expr]:
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
|
|
|
..\%tree_lua = nomsu:compile(\%tree)
|
2018-07-18 17:55:29 -07:00
|
|
|
if not \%tree_lua.is_value then
|
2018-08-28 15:08:00 -07:00
|
|
|
nomsu:compile_error(\%tree.source, "Could not convert %s to a Lua expression",
|
|
|
|
nomsu:tree_to_nomsu(\%tree))
|
2018-07-18 17:55:29 -07:00
|
|
|
end
|
2018-09-14 19:17:09 -07:00
|
|
|
return \%tree_lua"
|
2017-12-11 17:53:23 -08:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
2018-05-16 18:12:56 -07:00
|
|
|
|
2018-08-28 15:08:00 -07:00
|
|
|
compile [%tree as lua] to (Lua value "nomsu:compile(\(%tree as lua expr))")
|
2018-07-17 23:08:13 -07:00
|
|
|
compile [%tree as lua statements] to (..)
|
2018-08-28 15:08:00 -07:00
|
|
|
Lua value "nomsu:compile(\(%tree as lua expr)):as_statements()"
|
2018-05-16 18:12:56 -07:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
compile [%tree as lua return] to (..)
|
2018-08-28 15:08:00 -07:00
|
|
|
Lua value "nomsu:compile(\(%tree as lua expr)):as_statements('return ')"
|
2018-01-25 17:34:49 -08:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
compile [remove action %action] to (..)
|
2018-09-10 16:36:51 -07:00
|
|
|
Lua "\(=lua "(\(%action.stub)):as_lua_id()") = nil"
|
2018-01-25 17:34:49 -08:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
2018-09-14 19:17:09 -07:00
|
|
|
assume ("\(\(foo \%x) as nomsu)" == "foo %x") or barf "\
|
|
|
|
..action source code failed."
|
2018-07-17 23:08:13 -07:00
|
|
|
compile [%tree as nomsu] to (..)
|
2018-08-28 15:08:00 -07:00
|
|
|
Lua value "nomsu:tree_to_nomsu(\(%tree as lua expr))"
|
2018-06-12 13:56:15 -07:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
compile [%tree as inline nomsu] to (..)
|
2018-08-28 15:08:00 -07:00
|
|
|
Lua value "nomsu:tree_to_nomsu(\(%tree as lua expr), true)"
|
2018-06-14 21:59:25 -07:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
action [%var as lua identifier, %var as lua id] (..)
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
|
|
|
..if type(\%var) == 'string' then return \%var:as_lua_id()
|
2018-09-10 16:36:51 -07:00
|
|
|
elseif AST.is_syntax_tree(\%var, 'Var') then return \%var[1]:as_lua_id()
|
|
|
|
elseif AST.is_syntax_tree(\%var, 'Action') then return \%var.stub:as_lua_id()
|
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-07-17 23:08:13 -07:00
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
2018-06-26 15:52:38 -07:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
compile [% is syntax tree] to (Lua value "AST.is_syntax_tree(\(% as lua expr))")
|
|
|
|
compile [% is %kind syntax tree] to (..)
|
2018-06-26 15:52:38 -07:00
|
|
|
Lua value "AST.is_syntax_tree(\(% as lua expr), \(%kind as lua expr))"
|
2018-07-14 14:52:28 -07:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
compile [%tree with %t -> %replacement] to (..)
|
2018-09-14 19:17:09 -07:00
|
|
|
Lua value "\
|
|
|
|
..\(%tree as lua expr):map(function(\(%t as lua expr))
|
2018-06-18 15:44:29 -07:00
|
|
|
\(%replacement as lua return)
|
2018-09-14 19:17:09 -07:00
|
|
|
end)"
|
2018-01-25 17:34:49 -08:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
action [%tree with vars %replacements] (..)
|
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-08-27 13:38:58 -07:00
|
|
|
compile [tree %tree with vars %replacements] to (..)
|
2018-09-14 19:17:09 -07:00
|
|
|
Lua value "\
|
|
|
|
..\(=lua "repr(\%tree)"):map(function(t)
|
2018-08-27 13:38:58 -07:00
|
|
|
if t.type == "Var" then
|
|
|
|
return \(%replacements as lua expr)[t[1]]
|
|
|
|
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
|
|
|
compile [%tree has subtree %match_tree] to (..)
|
2018-09-14 19:17:09 -07:00
|
|
|
Lua value "\
|
|
|
|
..(function()
|
2018-07-22 13:59:08 -07:00
|
|
|
local match_tree = \(%match_tree as lua expr)
|
|
|
|
for subtree in coroutine.wrap(function() \(%tree as lua expr):map(coroutine.yield) end) do
|
|
|
|
if subtree == match_tree then return true end
|
|
|
|
end
|
2018-09-14 19:17:09 -07:00
|
|
|
end)()"
|
2018-07-22 13:59:08 -07:00
|
|
|
|
2018-08-27 13:38:58 -07:00
|
|
|
action [match %tree with %patt]:
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
|
|
|
..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-09-06 12:46:39 -07:00
|
|
|
local matches = _Dict{}
|
2018-08-27 13:38:58 -07:00
|
|
|
for \%i=1,#\%patt do
|
|
|
|
if AST.is_syntax_tree(\%tree[\%i]) then
|
|
|
|
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
|
|
|
|
|
|
|
action [%tree with %patt ~> %replacement]:
|
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-07-22 15:01:05 -07:00
|
|
|
assume (..)
|
|
|
|
(..)
|
2018-09-14 19:17:09 -07:00
|
|
|
quote "\
|
|
|
|
..one
|
|
|
|
"two""
|
2018-07-22 15:01:05 -07:00
|
|
|
..== "\"one\\n\\\"two\\\"\""
|
2018-07-17 23:08:13 -07:00
|
|
|
compile [quote %s] to (Lua value "repr(\(%s as lua expr))")
|
2018-07-30 15:05:41 -07:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
|
|
|
assume ((type of {}) == "table") or barf "type of failed."
|
2018-07-17 23:08:13 -07:00
|
|
|
compile [type of %obj] to (Lua value "type(\(%obj as lua expr))")
|
2018-07-22 13:59:08 -07:00
|
|
|
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
test:
|
2018-08-27 13:38:58 -07:00
|
|
|
assume ((parse "foo %") == \(foo \%))
|
2018-08-28 15:08:00 -07:00
|
|
|
%a = (parse "\\1")
|
2018-08-28 15:34:45 -07:00
|
|
|
%b = \(\(1))
|
|
|
|
assume ((parse "\\1") == \(\(1)))
|
2018-08-28 15:08:00 -07:00
|
|
|
compile [parse %text] to (Lua value "nomsu:parse(\(%text as lua expr))")
|
2018-07-17 23:08:13 -07:00
|
|
|
compile [parse %text from %filename] to (..)
|
2018-09-14 19:17:09 -07:00
|
|
|
Lua value "\
|
|
|
|
..nomsu:parse(NomsuCode(Source(\(%filename as lua expr), 1, #\(%text as lua expr)), \(..)
|
2018-07-20 20:27:15 -07:00
|
|
|
%text as lua expr
|
2018-09-14 19:17:09 -07:00
|
|
|
..))"
|
2018-06-26 15:52:38 -07:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
|
|
|
assume ((run "return (2 + 99)") == 101)
|
|
|
|
external %passed = (no)
|
2018-07-22 15:01:05 -07:00
|
|
|
run \:
|
2018-08-27 13:38:58 -07:00
|
|
|
\(external \%passed = \(yes))
|
2018-07-22 13:59:08 -07:00
|
|
|
assume %passed
|
2018-07-17 23:08:13 -07:00
|
|
|
compile [run %nomsu_code] to (..)
|
2018-09-14 19:17:09 -07:00
|
|
|
Lua value "\
|
|
|
|
..nomsu:run(\(%nomsu_code as lua expr), \(..)
|
2018-07-20 20:27:15 -07:00
|
|
|
=lua "repr(tostring(\(%nomsu_code.source)))"
|
2018-09-14 19:17:09 -07:00
|
|
|
..)"
|
2018-06-14 21:59:25 -07:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
2018-08-28 15:34:45 -07:00
|
|
|
assume ((\(\(5) + \(5)) as value) == 10) or barf "%tree as value failed."
|
2018-08-28 15:08:00 -07:00
|
|
|
action [run tree %tree, %tree as value] (lua> "return nomsu:run(\%tree)")
|
2018-07-17 23:08:13 -07:00
|
|
|
compile [compile %block, compiled %block, %block compiled] to (..)
|
2018-08-28 15:08:00 -07:00
|
|
|
Lua value "nomsu:compile(\(%block as lua))"
|
2018-01-25 17:34:49 -08:00
|
|
|
|
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-07-17 23:08:13 -07:00
|
|
|
compile [return] to (Lua "do return; end")
|
|
|
|
compile [return %return_value] to (..)
|
|
|
|
Lua "do return \(%return_value as lua expr) end"
|
|
|
|
|
2018-01-11 18:51:21 -08:00
|
|
|
# Literals
|
2018-07-17 23:08:13 -07:00
|
|
|
compile [yes] to (Lua value "true")
|
|
|
|
compile [no] to (Lua value "false")
|
|
|
|
compile [nothing, nil, null] to (Lua value "nil")
|
|
|
|
compile [Nomsu syntax version] to (Lua value "NOMSU_SYNTAX_VERSION")
|
|
|
|
compile [Nomsu compiler version] to (Lua value "NOMSU_COMPILER_VERSION")
|
|
|
|
compile [core version] to (Lua value "NOMSU_CORE_VERSION")
|
|
|
|
compile [lib version] to (Lua value "NOMSU_LIB_VERSION")
|
|
|
|
compile [command line args] to (Lua value "arg")
|
|
|
|
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
compile [with local compile actions %body] to (..)
|
2018-09-14 19:17:09 -07:00
|
|
|
Lua "\
|
|
|
|
..do
|
2018-08-27 13:38:58 -07:00
|
|
|
local nomsu = table.fork(nomsu, {COMPILE_ACTIONS=table.fork(COMPILE_ACTIONS)})
|
2018-06-26 15:52:38 -07:00
|
|
|
\(%body as lua statements)
|
2018-09-14 19:17:09 -07:00
|
|
|
end"
|
2018-06-26 15:52:38 -07:00
|
|
|
|
2018-07-17 23:08:13 -07:00
|
|
|
action [Nomsu version]:
|
2018-06-23 00:57:31 -07:00
|
|
|
use "lib/version.nom"
|
2018-09-14 19:17:09 -07:00
|
|
|
return "\
|
|
|
|
..\(Nomsu syntax version).\(core version).\(Nomsu compiler version).\(lib version)"
|