local lpeg = require('lpeg') local R, P, S R, P, S = lpeg.R, lpeg.P, lpeg.S local re = require('re') local List, Dict, Text do local _obj_0 = require('containers') List, Dict, Text = _obj_0.List, _obj_0.Dict, _obj_0.Text end local insert, remove, concat do local _obj_0 = table insert, remove, concat = _obj_0.insert, _obj_0.remove, _obj_0.concat end local unpack = unpack or table.unpack local match, sub, gsub, format, byte, find 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 end local LuaCode, Source do local _obj_0 = require("code_obj") LuaCode, Source = _obj_0.LuaCode, _obj_0.Source end local SyntaxTree = require("syntax_tree") local Importer, import_to_1_from, _1_forked do local _obj_0 = require('importer') Importer, import_to_1_from, _1_forked = _obj_0.Importer, _obj_0.import_to_1_from, _obj_0._1_forked end local Files = require("files") table.map = function(t, fn) return setmetatable((function() local _accum_0 = { } local _len_0 = 1 for _, v in ipairs(t) do _accum_0[_len_0] = fn(v) _len_0 = _len_0 + 1 end return _accum_0 end)(), getmetatable(t)) end local pretty_error = require("pretty_errors") local compile_error compile_error = function(source, err_msg, hint) if hint == nil then hint = nil end 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) end if source and not file then file = Files.read(source.filename) end local err_str = pretty_error({ title = "Compile error", error = err_msg, hint = hint, source = file, start = source.start, stop = source.stop, filename = source.filename }) return error(err_str, 0) end local math_expression = re.compile([[ (([*/^+-] / [0-9]+) " ")* [*/^+-] !. ]]) local MAX_LINE = 80 local compile = setmetatable({ action = Importer({ [""] = function(compile, fn, ...) local lua = LuaCode() local fn_lua = compile(fn) lua:append(fn_lua) if not (fn_lua:text():match("^%(.*%)$") or fn_lua:text():match("^[_a-zA-Z][_a-zA-Z0-9.]*$")) then lua:parenthesize() end lua:append("(") for i = 1, select('#', ...) do if i > 1 then lua:append(", ") end lua:append(compile((select(i, ...)))) end lua:append(")") return lua end, ["Lua"] = function(compile, code) if not code then return LuaCode("LuaCode()") end if code.type ~= "Text" then return LuaCode("LuaCode:from(", tostring(code.source):as_lua(), ", ", compile(code), ")") end local operate_on_text operate_on_text = function(text) local lua = LuaCode:from(text.source, "LuaCode:from(", tostring(text.source):as_lua()) for _index_0 = 1, #text do local bit = text[_index_0] local bit_lua if type(bit) == "string" then bit_lua = bit:as_lua() elseif bit.type == "Text" then bit_lua = operate_on_text(bit) elseif bit.type == "Block" then bit_lua = LuaCode:from(bit.source, "(function()", "\n local _lua = LuaCode:from(", tostring(bit.source):as_lua(), ")", "\n local function add(bit) _lua:append(bit) end", "\n local function join_with(glue)", "\n local old_bits = _lua.bits", "\n _lua = LuaCode:from(_lua.source)", "\n _lua:concat_append(old_bits, glue)", "\n end", "\n ", compile(bit), "\n return _lua", "\nend)()") else bit_lua = compile(bit) end local bit_leading_len = #(bit_lua:match("^[^\n]*")) lua:append(lua:trailing_line_len() + bit_leading_len > MAX_LINE and ",\n " or ", ") lua:append(bit_lua) end lua:append(")") return lua end return operate_on_text(code) end, ["lua >"] = function(compile, code) if code.type ~= "Text" then return code end local operate_on_text operate_on_text = function(text) local lua = LuaCode:from(text.source) for _index_0 = 1, #text do local bit = text[_index_0] if type(bit) == "string" then lua:append(bit) elseif bit.type == "Text" then lua:append(operate_on_text(bit)) else lua:append(compile(bit)) end end return lua end return operate_on_text(code) end, ["= lua"] = function(compile, code) return compile.action["lua >"](compile, code) end, ["use"] = function(compile, path) return LuaCode("run_file_1_in(" .. tostring(compile(path)) .. ", _ENV, OPTIMIZATION)") end, ["use 1 with prefix"] = function(compile, path, prefix) return LuaCode("run_file_1_in(" .. tostring(compile(path)) .. ", _ENV, OPTIMIZATION, ", compile(prefix), ")") end, ["tests"] = function(compile) return LuaCode("TESTS") end, ["test"] = function(compile, body) if not (body.type == 'Block') then compile_error(body, "This should be a Block") end local test_nomsu = body:get_source_code():match(":[ ]*(.*)") do local indent = test_nomsu:match("\n([ ]*)") if indent then test_nomsu = test_nomsu:gsub("\n" .. indent, "\n") end end return LuaCode("TESTS[" .. tostring(tostring(body.source):as_lua()) .. "] = ", test_nomsu:as_lua()) end, ["is jit"] = function(compile, code) return LuaCode("jit") end, ["Lua version"] = function(compile, code) return LuaCode("_VERSION") end, ["nomsu environment"] = function(compile) return LuaCode("_ENV") end }) }, { __import = import_to_1_from, __call = function(compile, tree) local _exp_0 = tree.type if "Action" == _exp_0 then local stub = tree.stub local compile_action = compile.action[stub] if not compile_action and not tree.target 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:append(tok) else local tok_lua = compile(tok) if tok.type == "Action" then tok_lua:parenthesize() end lua:append(tok_lua) end if i < #tree then lua:append(" ") end end return lua end if compile_action and not tree.target 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 end local ret = compile_action(compile, unpack(args)) if ret == nil then local info = debug.getinfo(compile_action, "S") local filename = Source:from_string(info.source).filename compile_error(tree, "The compile-time action here (" .. tostring(stub) .. ") failed to return any value.", "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 compile(ret) end end local lua = LuaCode:from(tree.source) if tree.target then local target_lua = compile(tree.target) local target_text = target_lua:text() if target_text:match("^%(.*%)$") or target_text:match("^[_a-zA-Z][_a-zA-Z0-9.]*$") or tree.target.type == "IndexChain" then lua:append(target_lua, ":") else lua:append("(", target_lua, "):") end end lua:append((stub):as_lua_id(), "(") local args = { } for i, tok in ipairs(tree) do local _continue_0 = false repeat if type(tok) == "string" then _continue_0 = true break end local arg_lua = compile(tok) if tok.type == "Block" then arg_lua = LuaCode:from(tok.source, "(function()\n ", arg_lua, "\nend)()") end insert(args, arg_lua) _continue_0 = true until true if not _continue_0 then break end end lua:concat_append(args, ", ") lua:append(")") 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 compile(x) 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 if needs_comma then lua:append(", ") else needs_comma = true end if k == i then i = i + 1 elseif type(k) == 'string' and match(k, "[_a-zA-Z][_a-zA-Z0-9]*") then lua:append(k, "= ") else lua:append("[", as_lua(k), "]= ") end if k == "source" then lua:append(tostring(v):as_lua()) else lua:append(as_lua(v)) end end lua:append("}") return lua elseif "Block" == _exp_0 then local lua = LuaCode:from(tree.source) for i, line in ipairs(tree) do if i > 1 then lua:append("\n") end lua:append(compile(line)) end return lua elseif "Text" == _exp_0 then local lua = LuaCode:from(tree.source) local added = 0 local string_buffer = "" for i, bit in ipairs(tree) do local _continue_0 = false repeat if type(bit) == "string" then string_buffer = string_buffer .. bit _continue_0 = true break end if string_buffer ~= "" then string_buffer = string_buffer:as_lua() if lua:trailing_line_len() + #string_buffer > MAX_LINE then lua:append("\n ") end if added > 0 then lua:append("..") end lua:append(string_buffer) added = added + 1 string_buffer = "" end local bit_lua = compile(bit) if bit.type == "Block" then bit_lua = LuaCode:from(bit.source, "(function()", "\n local _buffer = List{}", "\n local function add(bit) _buffer:add(bit) end", "\n local function join_with(glue) _buffer = _buffer:joined_with(glue) end", "\n ", bit_lua, "\n if lua_type_of(_buffer) == 'table' then _buffer = _buffer:joined() end", "\n return _buffer", "\nend)()") end if lua:trailing_line_len() + #bit_lua:text() > MAX_LINE then lua:append("\n ") end if added > 0 then lua:append("..") end if bit.type ~= "Text" then bit_lua = LuaCode:from(bit.source, "tostring(", bit_lua, ")") end lua:append(bit_lua) added = added + 1 _continue_0 = true until true if not _continue_0 then break end end if string_buffer ~= "" or #lua.bits == 0 then string_buffer = string_buffer:as_lua() if lua:trailing_line_len() + #string_buffer > MAX_LINE then lua:append("\n ") end if added > 0 then lua:append("..") end lua:append(string_buffer) added = added + 1 end if #lua.bits > 1 then lua:parenthesize() end return lua elseif "List" == _exp_0 or "Dict" == _exp_0 then local lua = LuaCode:from(tree.source) local i = 1 local sep = '' while i <= #tree do local item = tree[i] if item.type == "Block" then break end lua:append(sep) if item.type == "Comment" then lua:append(compile(item), "\n") sep = '' else local item_lua = compile(item) lua:append(item_lua) sep = ', ' end i = i + 1 end if lua:is_multiline() then lua = LuaCode:from(tree.source, tostring(tree.type) .. "{\n ", lua, "\n}") else lua = LuaCode:from(tree.source, tostring(tree.type) .. "{", lua, "}") end if i <= #tree then lua = LuaCode:from(tree.source, "(function()\n local comprehension = ", lua) if tree.type == "List" then lua:append("\n local function add(x) comprehension[#comprehension+1] = x end") else lua:append("\n local function " .. tostring(("add 1 ="):as_lua_id()) .. "(k, v) comprehension[k] = v end") end while i <= #tree do lua:append("\n ") if tree[i].type == 'Block' or tree[i].type == 'Comment' then lua:append(compile(tree[i])) elseif tree[i].type == "DictEntry" then local entry_lua = compile(tree[i]) lua:append((entry_lua:text():sub(1, 1) == '[' and "comprehension" or "comprehension."), entry_lua) else lua:append("comprehension[#comprehension+1] = ", compile(tree[i])) end i = i + 1 end lua:append("\n return comprehension\nend)()") end return lua elseif "DictEntry" == _exp_0 then local key, value = tree[1], tree[2] local key_lua = compile(key) local value_lua = value and compile(value) or LuaCode:from(key.source, "true") local key_str = match(key_lua:text(), [=[^["']([a-zA-Z_][a-zA-Z0-9_]*)['"]$]=]) if key_str and key_str:is_lua_id() then return LuaCode:from(tree.source, key_str, "=", value_lua) elseif sub(key_lua:text(), 1, 1) == "[" then return LuaCode:from(tree.source, "[ ", key_lua, "]=", value_lua) else return LuaCode:from(tree.source, "[", key_lua, "]=", value_lua) end elseif "IndexChain" == _exp_0 then local lua = compile(tree[1]) local first_char = sub(lua:text(), 1, 1) if first_char == "{" or first_char == '"' or first_char == "[" then lua:parenthesize() end for i = 2, #tree do local key = tree[i] local key_lua = compile(key) local key_lua_str = key_lua:text() local lua_id = match(key_lua_str, "^['\"]([a-zA-Z_][a-zA-Z0-9_]*)['\"]$") if lua_id and lua_id:is_lua_id() then lua:append("." .. tostring(lua_id)) elseif sub(key_lua_str, 1, 1) == '[' then lua:append("[ ", key_lua, " ]") else lua:append("[", key_lua, "]") end end return lua elseif "Number" == _exp_0 then return LuaCode:from(tree.source, tostring(tree[1])) elseif "Var" == _exp_0 then return LuaCode:from(tree.source, (concat(tree, " ")):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 return error("Can't compile errors") else return error("Unknown type: " .. tostring(tree.type)) end end }) return { compile = compile, compile_error = compile_error }