nomsu/nomsu_compiler.lua

465 lines
15 KiB
Lua
Raw Normal View History

2018-06-19 01:27:32 -07:00
local unpack = unpack or table.unpack
local match, sub, gsub, format, byte, find
2018-06-19 01:27:32 -07:00
do
local _obj_0 = string
match, sub, gsub, format, byte, find = _obj_0.match, _obj_0.sub, _obj_0.gsub, _obj_0.format, _obj_0.byte, _obj_0.find
2018-06-19 01:27:32 -07:00
end
local LuaCode, Source
2018-06-19 01:27:32 -07:00
do
local _obj_0 = require("code_obj")
LuaCode, Source = _obj_0.LuaCode, _obj_0.Source
2018-06-19 01:27:32 -07:00
end
require("text")
local SyntaxTree = require("syntax_tree")
local Files = require("files")
local pretty_error = require("pretty_errors")
local fail_at
fail_at = function(source, msg)
local file
if SyntaxTree:is_instance(source) then
file = source:get_source_file()
source = source.source
elseif type(source) == 'string' then
source = Source:from_string(source)
elseif not Source:is_instance(source) then
assert(source.short_src and source.currentline)
file = Files.read(source.short_src)
assert(file, "Could not find " .. tostring(source.short_src))
local lines = file:lines()
local start = 1
for i = 1, source.currentline - 1 do
start = start + #lines[i]
end
local stop = start + #lines[source.currentline]
source = Source(source.short_src, start, stop)
end
if source and not file then
file = Files.read(source.filename)
end
if not file then
if NOMSU_PREFIX then
local path = tostring(NOMSU_PREFIX) .. "/share/nomsu/" .. tostring(table.concat(NOMSU_VERSION, ".")) .. "/" .. tostring(source.filename)
file = Files.read(path)
end
end
if not file then
error("Can't find file: " .. tostring(source.filename))
end
local title, err_msg, hint = msg:match("([^:]*):[ \n]+(.*)[ \n]+Hint: (.*)")
if not err_msg then
err_msg, hint = msg:match("(.*)[ \n]+Hint:[ \n]+(.*)")
title = "Failure"
end
if not err_msg then
title, err_msg = msg:match("([^:]*):[ \n]+(.*)")
end
if not err_msg then
err_msg = msg
title = "Failure"
end
local err_str = pretty_error({
title = title,
error = err_msg,
hint = hint,
source = file,
start = source.start,
stop = source.stop,
filename = source.filename
2018-06-19 01:27:32 -07:00
})
return error(err_str, 0)
2018-06-19 01:27:32 -07:00
end
local re = require('re')
local math_expression = re.compile([[ (([*/^+-] / [0-9]+) " ")* [*/^+-] !. ]])
2018-06-19 01:27:32 -07:00
local MAX_LINE = 80
local compile
compile = function(self, tree)
if tree == nil then
error("No tree was passed in.")
end
if tree.version and tree.version < self.NOMSU_VERSION:up_to(#tree.version) and self._1_upgraded_from_2_to then
tree = self._1_upgraded_from_2_to(tree, tree.version, self.NOMSU_VERSION)
end
local _exp_0 = tree.type
if "Action" == _exp_0 then
local stub = tree.stub
local compile_action = self.COMPILE_RULES[stub]
if not compile_action and math_expression:match(stub) then
local lua = LuaCode:from(tree.source)
for i, tok in ipairs(tree) do
if type(tok) == 'string' then
lua:add(tok)
else
local tok_lua = self:compile(tok)
if tok.type == "Action" then
tok_lua:parenthesize()
end
lua:add(tok_lua)
end
if i < #tree then
lua:add(" ")
end
end
return lua
end
if not compile_action then
local seen_words = { }
local words = { }
for word in stub:gmatch("[^0-9 ][^ ]*") do
if not (seen_words[word]) then
seen_words[word] = true
table.insert(words, word)
end
end
table.sort(words)
local stub2 = table.concat(words, " ")
compile_action = self.COMPILE_RULES[stub2]
if compile_action then
if debug.getinfo(compile_action, 'u').isvararg then
stub = stub2
else
compile_action = nil
end
end
end
if compile_action then
local args
do
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1, #tree do
local arg = tree[_index_0]
if type(arg) ~= "string" then
_accum_0[_len_0] = arg
_len_0 = _len_0 + 1
end
end
args = _accum_0
2018-06-19 01:27:32 -07:00
end
local ret = compile_action(self, tree, unpack(args))
if ret == nil then
local info = debug.getinfo(compile_action, "S")
local filename = Source:from_string(info.source).filename
fail_at(tree, ("Compile error: The compile-time action here (" .. tostring(stub) .. ") failed to return any value. " .. "Hint: Look at the implementation of (" .. tostring(stub) .. ") in " .. tostring(filename) .. ":" .. tostring(info.linedefined) .. " and make sure it's returning something."))
2018-06-19 01:27:32 -07:00
end
if not (SyntaxTree:is_instance(ret)) then
ret.source = ret.source or tree.source
return ret
2018-06-19 01:27:32 -07:00
end
if ret ~= tree then
return self:compile(ret)
end
2018-06-19 01:27:32 -07:00
end
local lua = LuaCode:from(tree.source)
lua:add((stub):as_lua_id(), "(")
for argnum, arg in ipairs(tree:get_args()) do
local arg_lua = self:compile(arg)
if arg.type == "Block" and #arg > 1 then
arg_lua = LuaCode:from(arg.source, "(function()\n ", arg_lua, "\nend)()")
end
if lua:trailing_line_len() + #arg_lua > MAX_LINE then
lua:add(argnum > 1 and ",\n " or "\n ")
elseif argnum > 1 then
lua:add(", ")
2018-06-19 01:27:32 -07:00
end
lua:add(arg_lua)
end
lua:add(")")
return lua
elseif "MethodCall" == _exp_0 then
local stub = tree:get_stub()
local compile_action = self.COMPILE_RULES[stub]
if compile_action then
local args = tree:get_args()
local ret = compile_action(self, tree, unpack(args))
if ret == nil then
local info = debug.getinfo(compile_action, "S")
local filename = Source:from_string(info.source).filename
fail_at(tree, ("Compile error: The compile-time method here (" .. tostring(stub) .. ") failed to return any value. " .. "Hint: Look at the implementation of (" .. tostring(stub) .. ") in " .. tostring(filename) .. ":" .. tostring(info.linedefined) .. " " .. "and make sure it's returning something."))
end
if not (SyntaxTree:is_instance(ret)) then
ret.source = ret.source or tree.source
return ret
end
if ret ~= tree then
return self:compile(ret)
end
end
local lua = LuaCode:from(tree.source)
local target_lua = self:compile(tree[1])
local target_text = target_lua:text()
if not (target_text:match("^%(.*%)$") or target_text:match("^[_a-zA-Z][_a-zA-Z0-9.]*$") or tree[1].type == "IndexChain") then
target_lua:parenthesize()
end
local self_lua = #tree > 2 and "_self" or target_lua
if #tree > 2 then
lua:add("(function(", self_lua, ")\n ")
end
for i = 2, #tree do
if i > 2 then
lua:add("\n ")
end
if i > 2 and i == #tree then
lua:add("return ")
end
lua:add(self_lua, ":")
lua:add((tree[i].stub):as_lua_id(), "(")
for argnum, arg in ipairs(tree[i]:get_args()) do
local arg_lua = self:compile(arg)
if arg.type == "Block" and #arg > 1 then
arg_lua = LuaCode:from(arg.source, "(function()\n ", arg_lua, "\nend)()")
end
if lua:trailing_line_len() + #arg_lua > MAX_LINE then
2018-12-14 19:25:59 -08:00
lua:add(argnum > 1 and ",\n " or "\n ")
elseif argnum > 1 then
lua:add(", ")
end
lua:add(arg_lua)
end
lua:add(")")
end
if #tree > 2 then
lua:add("\nend)(", target_lua, ")")
end
return lua
elseif "EscapedNomsu" == _exp_0 then
local lua = LuaCode:from(tree.source, "SyntaxTree{")
local needs_comma, i = false, 1
local as_lua
as_lua = function(x)
if type(x) == 'number' then
return tostring(x)
elseif SyntaxTree:is_instance(x) then
return self:compile(x)
elseif Source:is_instance(x) then
return tostring(x):as_lua()
else
return x:as_lua()
end
end
for k, v in pairs((SyntaxTree:is_instance(tree[1]) and tree[1].type == "EscapedNomsu" and tree) or tree[1]) do
local entry_lua = LuaCode()
if k == i then
i = i + 1
elseif type(k) == 'string' and match(k, "[_a-zA-Z][_a-zA-Z0-9]*") then
entry_lua:add(k, "= ")
else
entry_lua:add("[", as_lua(k), "]= ")
2018-06-19 01:27:32 -07:00
end
entry_lua:add(as_lua(v))
if needs_comma then
lua:add(",")
end
if lua:trailing_line_len() + #(entry_lua:match("^[\n]*")) > MAX_LINE then
lua:add("\n ")
elseif needs_comma then
lua:add(" ")
2018-06-19 01:27:32 -07:00
end
lua:add(entry_lua)
needs_comma = true
end
lua:add("}")
return lua
elseif "Block" == _exp_0 then
local lua = LuaCode:from(tree.source)
for i, line in ipairs(tree) do
if line.type == "Error" then
return self:compile(line)
end
end
for i, line in ipairs(tree) do
if i > 1 then
lua:add("\n")
end
local line_lua = self:compile(line)
lua:add(line_lua)
2019-03-13 20:53:30 -07:00
if not (line_lua:last(1) == ";" or line_lua:last(4):match("[^_a-zA-Z0-9]end$")) then
lua:add(";")
end
end
return lua
elseif "Text" == _exp_0 then
if #tree == 0 then
return LuaCode:from(tree.source, '""')
end
if #tree == 1 and type(tree[1]) == 'string' then
return LuaCode:from(tree.source, tree[1]:as_lua())
end
local lua = LuaCode:from(tree.source, "Text(")
local added = 0
local string_buffer = ""
local add_bit
add_bit = function(bit)
if added > 0 then
if lua:trailing_line_len() + #bit > MAX_LINE then
lua:add(",\n ")
else
lua:add(", ")
end
end
lua:add(bit)
added = added + 1
end
for i, bit in ipairs(tree) do
local _continue_0 = false
repeat
if type(bit) == "string" then
string_buffer = string_buffer .. bit
2018-06-19 01:27:32 -07:00
_continue_0 = true
break
end
if string_buffer ~= "" then
for i = 1, #string_buffer, MAX_LINE do
add_bit(string_buffer:sub(i, i + MAX_LINE - 1):as_lua())
end
string_buffer = ""
2018-06-19 01:27:32 -07:00
end
local bit_lua = self:compile(bit)
if bit.type == "Block" then
bit_lua = LuaCode:from(bit.source, "a_List(function(add)", "\n ", bit_lua, "\nend):joined()")
end
add_bit(bit_lua)
_continue_0 = true
until true
if not _continue_0 then
break
2018-06-19 01:27:32 -07:00
end
end
if string_buffer ~= "" then
for i = 1, #string_buffer, MAX_LINE do
add_bit(string_buffer:sub(i, i + MAX_LINE - 1):as_lua())
2018-12-18 19:30:01 -08:00
end
string_buffer = ""
end
if added == 0 then
return LuaCode:from(tree.source, '""')
end
lua:add(")")
return lua
elseif "List" == _exp_0 or "Dict" == _exp_0 then
local typename = "a_" .. tree.type
if #tree == 0 then
return LuaCode:from(tree.source, typename, "{}")
end
local lua = LuaCode:from(tree.source)
local chunks = 0
local i = 1
while tree[i] do
if tree[i].type == 'Block' then
if chunks > 0 then
lua:add(" + ")
end
lua:add(typename, "(function(", (tree.type == 'List' and "add" or ("add, " .. ("add 1 ="):as_lua_id())), ")")
local body = self:compile(tree[i])
body:declare_locals()
lua:add("\n ", body, "\nend)")
chunks = chunks + 1
i = i + 1
else
if chunks > 0 then
lua:add(" + ")
end
local sep = ''
local items_lua = LuaCode:from(tree[i].source)
while tree[i] do
if tree[i].type == "Block" then
break
2018-12-18 19:30:01 -08:00
end
local item_lua = self:compile(tree[i])
if item_lua:match("^%.[a-zA-Z_]") then
item_lua = item_lua:text():sub(2)
2018-12-18 19:30:01 -08:00
end
if tree.type == 'Dict' and tree[i].type == 'Index' then
item_lua = LuaCode:from(tree[i].source, item_lua, "=true")
2018-12-18 19:30:01 -08:00
end
items_lua:add(sep, item_lua)
if tree[i].type == "Comment" then
items_lua:add("\n")
sep = ''
elseif items_lua:trailing_line_len() > MAX_LINE then
sep = items_lua:last(1) == ";" and "\n " or ",\n "
else
sep = items_lua:last(1) == ";" and " " or ", "
end
i = i + 1
end
if items_lua:is_multiline() then
lua:add(LuaCode:from(items_lua.source, typename, "{\n ", items_lua, "\n}"))
else
lua:add(LuaCode:from(items_lua.source, typename, "{", items_lua, "}"))
end
chunks = chunks + 1
end
end
return lua
elseif "Index" == _exp_0 then
local key_lua = self:compile(tree[1])
local key_str = key_lua:match('^"([a-zA-Z_][a-zA-Z0-9_]*)"$')
if key_str and key_str:is_lua_id() then
return LuaCode:from(tree.source, ".", key_str)
elseif key_lua:first(1) == "[" then
return LuaCode:from(tree.source, "[ ", key_lua, "]")
else
return LuaCode:from(tree.source, "[", key_lua, "]")
end
elseif "DictEntry" == _exp_0 then
local key = tree[1]
if key.type ~= "Index" then
key = SyntaxTree({
type = "Index",
source = key.source,
key
})
end
return LuaCode:from(tree.source, self:compile(key), "=", (tree[2] and self:compile(tree[2]) or "true"))
elseif "IndexChain" == _exp_0 then
local lua = self:compile(tree[1])
if lua:match("['\"}]$") or lua:match("]=*]$") then
lua:parenthesize()
end
if lua:text() == "..." then
return LuaCode:from(tree.source, "select(", self:compile(tree[2][1]), ", ...)")
end
for i = 2, #tree do
local key = tree[i]
2018-12-18 19:30:01 -08:00
if key.type ~= "Index" then
key = SyntaxTree({
type = "Index",
source = key.source,
key
})
end
lua:add(self:compile(key))
2018-06-19 01:27:32 -07:00
end
return lua
elseif "Number" == _exp_0 then
local number = tostring(tree[1]):gsub("_", "")
return LuaCode:from(tree.source, number)
elseif "Var" == _exp_0 then
if tree[1].type == "MethodCall" then
return LuaCode:from(tree.source, self:compile(tree[1][1]), ".", tree[1][2]:get_stub():as_lua_id())
end
return LuaCode:from(tree.source, tree:as_var():as_lua_id())
elseif "FileChunks" == _exp_0 then
return error("Can't convert FileChunks to a single block of lua, since each chunk's " .. "compilation depends on the earlier chunks")
elseif "Comment" == _exp_0 then
return LuaCode:from(tree.source, "-- ", (tree[1]:gsub('\n', '\n-- ')))
elseif "Error" == _exp_0 then
local err_msg = pretty_error({
title = "Parse error",
error = tree.error,
hint = tree.hint,
source = tree:get_source_file(),
start = tree.source.start,
stop = tree.source.stop,
filename = tree.source.filename
})
return error(err_msg)
else
return error("Unknown type: " .. tostring(tree.type))
2018-06-19 01:27:32 -07:00
end
end
return {
compile = compile,
fail_at = fail_at
}