From 59c79bdf57ce31d5f2fe15dbbdcc6ff345adc651 Mon Sep 17 00:00:00 2001 From: Bruce Hill Date: Tue, 19 Jun 2018 01:27:32 -0700 Subject: [PATCH] Split up CLI and NomsuCompiler --- nomsu.lua | 1470 +++++-------------------------------------- nomsu.moon | 1063 ++++--------------------------- nomsu_compiler.lua | 1148 +++++++++++++++++++++++++++++++++ nomsu_compiler.moon | 797 +++++++++++++++++++++++ 4 files changed, 2246 insertions(+), 2232 deletions(-) create mode 100644 nomsu_compiler.lua create mode 100644 nomsu_compiler.moon diff --git a/nomsu.lua b/nomsu.lua index 1f53102..b500cfa 100644 --- a/nomsu.lua +++ b/nomsu.lua @@ -1,1173 +1,4 @@ -local lpeg = require('lpeg') -local re = require('re') -lpeg.setmaxstack(10000) -local utils = require('utils') -local repr, stringify, equivalent -repr, stringify, equivalent = utils.repr, utils.stringify, utils.equivalent -colors = require('consolecolors') -colored = setmetatable({ }, { - __index = function(_, color) - return (function(msg) - return colors[color] .. tostring(msg or '') .. colors.reset - end) - 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, rep, gsub, format, byte, find -do - local _obj_0 = string - match, sub, rep, gsub, format, byte, match, find = _obj_0.match, _obj_0.sub, _obj_0.rep, _obj_0.gsub, _obj_0.format, _obj_0.byte, _obj_0.match, _obj_0.find -end -local NomsuCode, LuaCode, Source -do - local _obj_0 = require("code_obj") - NomsuCode, LuaCode, Source = _obj_0.NomsuCode, _obj_0.LuaCode, _obj_0.Source -end -local AST = require("nomsu_tree") -local parse = require("parser") -local STDIN, STDOUT, STDERR = "/dev/fd/0", "/dev/fd/1", "/dev/fd/2" -SOURCE_MAP = { } -string.as_lua_id = function(str) - local argnum = 0 - str = gsub(str, "x([0-9A-F][0-9A-F])", "x\0%1") - str = gsub(str, "%W", function(c) - if c == ' ' then - return '_' - elseif c == '%' then - argnum = argnum + 1 - return tostring(argnum) - else - return format("x%02X", byte(c)) - end - end) - return '_' .. str -end -table.map = function(self, fn) - local _accum_0 = { } - local _len_0 = 1 - for _, v in ipairs(self) do - _accum_0[_len_0] = fn(v) - _len_0 = _len_0 + 1 - end - return _accum_0 -end -table.fork = function(t, values) - return setmetatable(values or { }, { - __index = t - }) -end -FILE_CACHE = setmetatable({ }, { - __index = function(self, filename) - local file = io.open(filename) - if not (file) then - return nil - end - local contents = file:read("*a") - file:close() - self[filename] = contents - return contents - end -}) -local iterate_single -iterate_single = function(item, prev) - if item == prev then - return nil - else - return item - end -end -local all_files -all_files = function(path) - if match(path, "%.nom$") or match(path, "%.lua$") or match(path, "^/dev/fd/[012]$") then - return iterate_single, path - end - path = gsub(path, "\\", "\\\\") - path = gsub(path, "`", "") - path = gsub(path, '"', '\\"') - path = gsub(path, "$", "") - return coroutine.wrap(function() - local f = io.popen('find -L "' .. path .. '" -not -path "*/\\.*" -type f -name "*.nom"') - for line in f:lines() do - coroutine.yield(line) - end - local success = f:close() - if not (success) then - return error("Invalid file path: " .. tostring(path)) - end - end) -end -local line_counter = re.compile([[ lines <- {| line (%nl line)* |} - line <- {} (!%nl .)* -]], { - nl = lpeg.P("\r") ^ -1 * lpeg.P("\n") -}) -local get_lines = re.compile([[ lines <- {| line (%nl line)* |} - line <- {[^%nl]*} -]], { - nl = lpeg.P("\r") ^ -1 * lpeg.P("\n") -}) -LINE_STARTS = setmetatable({ }, { - __mode = "k", - __index = function(self, k) - if type(k) ~= 'string' then - k = tostring(k) - do - local v = rawget(self, k) - if v then - return v - end - end - end - local line_starts = line_counter:match(k) - self[k] = line_starts - return line_starts - end -}) -pos_to_line = function(str, pos) - local line_starts = LINE_STARTS[str] - local lo, hi = 1, #line_starts - while lo <= hi do - local mid = math.floor((lo + hi) / 2) - if line_starts[mid] > pos then - hi = mid - 1 - else - lo = mid + 1 - end - end - return hi -end -do - local STRING_METATABLE = getmetatable("") - STRING_METATABLE.__add = function(self, other) - return self .. stringify(other) - end - STRING_METATABLE.__index = function(self, i) - local ret = string[i] - if ret ~= nil then - return ret - end - if type(i) == 'number' then - return sub(self, i, i) - elseif type(i) == 'table' then - return sub(self, i[1], i[2]) - end - end -end -local _list_mt = { - __eq = equivalent, - __tostring = function(self) - return "[" .. concat((function() - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #self do - local b = self[_index_0] - _accum_0[_len_0] = repr(b) - _len_0 = _len_0 + 1 - end - return _accum_0 - end)(), ", ") .. "]" - end -} -local list -list = function(t) - return setmetatable(t, _list_mt) -end -local _dict_mt = { - __eq = equivalent, - __tostring = function(self) - return "{" .. concat((function() - local _accum_0 = { } - local _len_0 = 1 - for k, v in pairs(self) do - _accum_0[_len_0] = tostring(repr(k)) .. ": " .. tostring(repr(v)) - _len_0 = _len_0 + 1 - end - return _accum_0 - end)(), ", ") .. "}" - end -} -local dict -dict = function(t) - return setmetatable(t, _dict_mt) -end -local MAX_LINE = 80 -local NomsuCompiler = setmetatable({ }, { - __index = function(self, k) - do - local _self = rawget(self, "self") - if _self then - return _self[k] - else - return nil - end - end - end -}) -do - NomsuCompiler._ENV = NomsuCompiler - NomsuCompiler.nomsu = NomsuCompiler - NomsuCompiler.parse = function(self, ...) - return parse(...) - end - local to_add = { - repr = repr, - stringify = stringify, - utils = utils, - lpeg = lpeg, - re = re, - next = next, - unpack = unpack, - setmetatable = setmetatable, - coroutine = coroutine, - rawequal = rawequal, - getmetatable = getmetatable, - pcall = pcall, - error = error, - package = package, - os = os, - require = require, - tonumber = tonumber, - tostring = tostring, - string = string, - xpcall = xpcall, - module = module, - print = print, - loadfile = loadfile, - rawset = rawset, - _VERSION = _VERSION, - collectgarbage = collectgarbage, - rawget = rawget, - rawlen = rawlen, - table = table, - assert = assert, - dofile = dofile, - loadstring = loadstring, - type = type, - select = select, - debug = debug, - math = math, - io = io, - load = load, - pairs = pairs, - ipairs = ipairs, - list = list, - dict = dict - } - for k, v in pairs(to_add) do - NomsuCompiler[k] = v - end - for k, v in pairs(AST) do - NomsuCompiler[k] = v - end - NomsuCompiler.LuaCode = LuaCode - NomsuCompiler.NomsuCode = NomsuCode - NomsuCompiler.Source = Source - NomsuCompiler.ALIASES = setmetatable({ }, { - __mode = "k" - }) - NomsuCompiler.LOADED = { } - NomsuCompiler.AST = AST - NomsuCompiler.compile_error = function(self, tok, err_format_string, ...) - local file = FILE_CACHE[tok.source.filename] - local line_no = pos_to_line(file, tok.source.start) - local line_start = LINE_STARTS[file][line_no] - local src = colored.dim(file:sub(line_start, tok.source.start - 1)) - src = src .. colored.underscore(colored.bright(colored.red(file:sub(tok.source.start, tok.source.stop - 1)))) - local end_of_line = (LINE_STARTS[file][pos_to_line(file, tok.source.stop) + 1] or 0) - 1 - src = src .. colored.dim(file:sub(tok.source.stop, end_of_line - 1)) - src = ' ' .. src:gsub('\n', '\n ') - local err_msg = err_format_string:format(src, ...) - return error(tostring(tok.source.filename) .. ":" .. tostring(line_no) .. ": " .. err_msg, 0) - end - local math_expression = re.compile([[ ([+-] " ")* "%" (" " [*/^+-] (" " [+-])* " %")+ !. ]]) - local add_lua_bits - add_lua_bits = function(self, lua, code) - for _index_0 = 1, #code do - local bit = code[_index_0] - if type(bit) == "string" then - lua:append(bit) - else - local bit_lua = self:compile(bit) - if not (bit_lua.is_value) then - self:compile_error(bit, "Cannot use:\n%s\nas a string interpolation value, since it's not an expression.") - end - lua:append(bit_lua) - end - end - return lua - end - local add_lua_string_bits - add_lua_string_bits = function(self, lua, code) - local line_len = 0 - if code.type ~= "Text" then - lua:append(", ", self:compile(code)) - return - end - for _index_0 = 1, #code do - local bit = code[_index_0] - local bit_lua - if type(bit) == "string" then - bit_lua = repr(bit) - else - bit_lua = self:compile(bit) - if not (bit_lua.is_value) then - self:compile_error(bit, "Cannot use:\n%s\nas a string interpolation value, since it's not an expression.") - end - bit_lua = bit_lua - end - line_len = line_len + #tostring(bit_lua) - if line_len > MAX_LINE then - lua:append(",\n ") - line_len = 4 - else - lua:append(", ") - end - lua:append(bit_lua) - end - end - NomsuCompiler.COMPILE_ACTIONS = setmetatable({ - ["# compile math expr #"] = function(self, tree, ...) - local lua = LuaCode.Value(tree.source) - for i, tok in ipairs(tree) do - if type(tok) == 'string' then - lua:append(tok) - else - local tok_lua = self:compile(tok) - if not (tok_lua.is_value) then - self:compile_error(tok, "Non-expression value inside math expression:\n%s") - end - 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, - ["Lua %"] = function(self, tree, _code) - local lua = LuaCode.Value(_code.source, "LuaCode(", repr(tostring(_code.source))) - add_lua_string_bits(self, lua, _code) - lua:append(")") - return lua - end, - ["Lua value %"] = function(self, tree, _code) - local lua = LuaCode.Value(_code.source, "LuaCode.Value(", repr(tostring(_code.source))) - add_lua_string_bits(self, lua, _code) - lua:append(")") - return lua - end, - ["lua > %"] = function(self, tree, _code) - if _code.type ~= "Text" then - return LuaCode(tree.source, "nomsu:run_lua(", self:compile(_code), ");") - end - return add_lua_bits(self, LuaCode(tree.source), _code) - end, - ["= lua %"] = function(self, tree, _code) - if _code.type ~= "Text" then - return LuaCode.Value(tree.source, "nomsu:run_lua(", self:compile(_code), ":as_statements('return '))") - end - return add_lua_bits(self, LuaCode.Value(tree.source), _code) - end, - ["use %"] = function(self, tree, _path) - if not (_path.type == 'Text' and #_path == 1 and type(_path[1]) == 'string') then - return LuaCode(tree.source, "nomsu:run_file(" .. tostring(self:compile(_path)) .. ");") - end - local path = _path[1] - self:run_file(path) - return LuaCode(tree.source, "nomsu:run_file(" .. tostring(repr(path)) .. ");") - end - }, { - __index = function(self, stub) - if math_expression:match(stub) then - return self["# compile math expr #"] - end - end - }) - NomsuCompiler.run = function(self, to_run, source) - if source == nil then - source = nil - end - local tree - if AST.is_syntax_tree(to_run) then - tree = tree - else - tree = self:parse(to_run, source or to_run.source) - end - if tree == nil then - return nil - end - if tree.type == "FileChunks" then - local ret = nil - local all_lua = { } - for _index_0 = 1, #tree do - local chunk = tree[_index_0] - local lua = self:compile(chunk):as_statements() - lua:declare_locals() - lua:prepend("-- File: " .. tostring(chunk.source or "") .. "\n") - insert(all_lua, tostring(lua)) - ret = self:run_lua(lua) - end - if self.on_compile then - self.on_compile(concat(all_lua, "\n"), (source or to_run.source).filename) - end - return ret - else - local lua = self:compile(tree, compile_actions):as_statements() - lua:declare_locals() - lua:prepend("-- File: " .. tostring(source or to_run.source or "") .. "\n") - if self.on_compile then - self.on_compile(lua, (source or to_run.source).filename) - end - return self:run_lua(lua) - end - end - local _running_files = { } - NomsuCompiler.run_file = function(self, filename) - if self.LOADED[filename] then - return self.LOADED[filename] - end - local ret = nil - for filename in all_files(filename) do - local _continue_0 = false - repeat - do - ret = self.LOADED[filename] - if ret then - _continue_0 = true - break - end - end - for i, running in ipairs(_running_files) do - if running == filename then - local loop - do - local _accum_0 = { } - local _len_0 = 1 - for j = i, #_running_files do - _accum_0[_len_0] = _running_files[j] - _len_0 = _len_0 + 1 - end - loop = _accum_0 - end - insert(loop, filename) - error("Circular import, this loops forever: " .. tostring(concat(loop, " -> ")) .. "...") - end - end - insert(_running_files, filename) - if match(filename, "%.lua$") then - local file = assert(FILE_CACHE[filename], "Could not find file: " .. tostring(filename)) - ret = self:run_lua(file, Source(filename, 1, #file)) - elseif match(filename, "%.nom$") or match(filename, "^/dev/fd/[012]$") then - local ran_lua - if not self.skip_precompiled then - local lua_filename = gsub(filename, "%.nom$", ".lua") - do - local file = FILE_CACHE[lua_filename] - if file then - ret = self:run_lua(file, Source(lua_filename, 1, #file)) - ran_lua = true - end - end - end - if not (ran_lua) then - local file = file or FILE_CACHE[filename] - if not file then - error("File does not exist: " .. tostring(filename), 0) - end - ret = self:run(file, Source(filename, 1, #file)) - end - else - error("Invalid filetype for " .. tostring(filename), 0) - end - self.LOADED[filename] = ret or true - remove(_running_files) - _continue_0 = true - until true - if not _continue_0 then - break - end - end - self.LOADED[filename] = ret or true - return ret - end - NomsuCompiler.run_lua = function(self, lua, source) - if source == nil then - source = nil - end - local lua_string = tostring(lua) - local run_lua_fn, err = load(lua_string, nil and tostring(source or lua.source), "t", self) - if not run_lua_fn then - local line_numbered_lua = concat((function() - local _accum_0 = { } - local _len_0 = 1 - for i, line in ipairs(get_lines:match(lua_string)) do - _accum_0[_len_0] = format("%3d|%s", i, line) - _len_0 = _len_0 + 1 - end - return _accum_0 - end)(), "\n") - error("Failed to compile generated code:\n" .. tostring(colored.bright(colored.blue(colored.onblack(line_numbered_lua)))) .. "\n\n" .. tostring(err), 0) - end - local source_key = tostring(source or lua.source) - if not (SOURCE_MAP[source_key]) then - local map = { } - local offset = 1 - source = source or lua.source - local nomsu_str = tostring(FILE_CACHE[source.filename]:sub(source.start, source.stop)) - local lua_line = 1 - local nomsu_line = pos_to_line(nomsu_str, source.start) - local fn - fn = function(s) - if type(s) == 'string' then - for nl in s:gmatch("\n") do - map[lua_line] = map[lua_line] or nomsu_line - lua_line = lua_line + 1 - end - else - local old_line = nomsu_line - if s.source then - nomsu_line = pos_to_line(nomsu_str, s.source.start) - end - local _list_0 = s.bits - for _index_0 = 1, #_list_0 do - local b = _list_0[_index_0] - fn(b) - end - end - end - fn(lua) - map[lua_line] = map[lua_line] or nomsu_line - map[0] = 0 - SOURCE_MAP[source_key] = map - end - return run_lua_fn() - end - NomsuCompiler.compile = function(self, tree) - local _exp_0 = tree.type - if "Action" == _exp_0 then - local stub = tree.stub - do - local compile_action = self.COMPILE_ACTIONS[stub] - 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 - end - local ret = compile_action(self, tree, unpack(args)) - if not ret then - self:compile_error(tree, "Compile-time action:\n%s\nfailed to produce any Lua") - end - return ret - end - end - local lua = LuaCode.Value(tree.source, "A", string.as_lua_id(stub), "(") - 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 = self:compile(tok) - if not (arg_lua.is_value) then - self:compile_error(tok, "Cannot use:\n%s\nas an argument to %s, since it's not an expression, it produces: %s", stub, repr(arg_lua)) - 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 make_tree - make_tree = function(t) - if not (AST.is_syntax_tree(t)) then - return repr(t) - end - local bits - do - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #t do - local bit = t[_index_0] - _accum_0[_len_0] = make_tree(bit) - _len_0 = _len_0 + 1 - end - bits = _accum_0 - end - return t.type .. "(" .. repr(tostring(t.source)) .. ", " .. table.concat(bits, ", ") .. ")" - end - return LuaCode.Value(tree.source, make_tree(tree[1])) - elseif "Block" == _exp_0 then - local lua = LuaCode(tree.source) - lua:concat_append((function() - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #tree do - local line = tree[_index_0] - _accum_0[_len_0] = self:compile(line):as_statements() - _len_0 = _len_0 + 1 - end - return _accum_0 - end)(), "\n") - return lua - elseif "Text" == _exp_0 then - local lua = LuaCode.Value(tree.source) - 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 - if #lua.bits > 0 then - lua:append("..") - end - lua:append(repr(string_buffer)) - string_buffer = "" - end - local bit_lua = self:compile(bit) - if not (bit_lua.is_value) then - local src = ' ' .. gsub(tostring(self:tree_to_nomsu(bit)), '\n', '\n ') - local line = tostring(bit.source.filename) .. ":" .. tostring(pos_to_line(FILE_CACHE[bit.source.filename], bit.source.start)) - self:compile_error(bit, "Cannot use:\n%s\nas a string interpolation value, since it's not an expression.") - end - if #lua.bits > 0 then - lua:append("..") - end - if bit.type ~= "Text" then - bit_lua = LuaCode.Value(bit.source, "stringify(", bit_lua, ")") - end - lua:append(bit_lua) - _continue_0 = true - until true - if not _continue_0 then - break - end - end - if string_buffer ~= "" or #lua.bits == 0 then - if #lua.bits > 0 then - lua:append("..") - end - lua:append(repr(string_buffer)) - end - if #lua.bits > 1 then - lua:parenthesize() - end - return lua - elseif "List" == _exp_0 then - local lua = LuaCode.Value(tree.source, "list{") - local items = { } - for i, item in ipairs(tree) do - local item_lua = self:compile(item) - if not (item_lua.is_value) then - self:compile_error(item, "Cannot use:\n%s\nas a list item, since it's not an expression.") - end - items[i] = item_lua - end - lua:concat_append(items, ", ", ",\n ") - lua:append("}") - return lua - elseif "Dict" == _exp_0 then - local lua = LuaCode.Value(tree.source, "dict{") - lua:concat_append((function() - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #tree do - local e = tree[_index_0] - _accum_0[_len_0] = self:compile(e) - _len_0 = _len_0 + 1 - end - return _accum_0 - end)(), ", ", ",\n ") - lua:append("}") - return lua - elseif "DictEntry" == _exp_0 then - local key, value = tree[1], tree[2] - local key_lua = self:compile(key) - if not (key_lua.is_value) then - self:compile_error(tree[1], "Cannot use:\n%s\nas a dict key, since it's not an expression.") - end - local value_lua = value and self:compile(value) or LuaCode.Value(key.source, "true") - if not (value_lua.is_value) then - self:compile_error(tree[2], "Cannot use:\n%s\nas a dict value, since it's not an expression.") - end - local key_str = match(tostring(key_lua), [=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=]) - if key_str then - return LuaCode(tree.source, key_str, "=", value_lua) - elseif sub(tostring(key_lua), 1, 1) == "[" then - return LuaCode(tree.source, "[ ", key_lua, "]=", value_lua) - else - return LuaCode(tree.source, "[", key_lua, "]=", value_lua) - end - elseif "IndexChain" == _exp_0 then - local lua = self:compile(tree[1]) - if not (lua.is_value) then - self:compile_error(tree[1], "Cannot index:\n%s\nsince it's not an expression.") - end - local first_char = sub(tostring(lua), 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 = self:compile(key) - if not (key_lua.is_value) then - self:compile_error(key, "Cannot use:\n%s\nas an index, since it's not an expression.") - end - local key_lua_str = tostring(key_lua) - do - local lua_id = match(key_lua_str, "^['\"]([a-zA-Z_][a-zA-Z0-9_]*)['\"]$") - if 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 - end - return lua - elseif "Number" == _exp_0 then - return LuaCode.Value(tree.source, tostring(tree[1])) - elseif "Var" == _exp_0 then - return LuaCode.Value(tree.source, string.as_lua_id(tree[1])) - elseif "FileChunks" == _exp_0 then - return error("Cannot convert FileChunks to a single block of lua, since each chunk's " .. "compilation depends on the earlier chunks") - else - return error("Unknown type: " .. tostring(tree.type)) - end - end - NomsuCompiler.tree_to_nomsu = function(self, tree, inline, can_use_colon) - if inline == nil then - inline = false - end - if can_use_colon == nil then - can_use_colon = false - end - local _exp_0 = tree.type - if "FileChunks" == _exp_0 then - if inline then - return nil - end - local nomsu = NomsuCode(tree.source) - nomsu:concat_append((function() - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #tree do - local c = tree[_index_0] - _accum_0[_len_0] = self:tree_to_nomsu(c) - _len_0 = _len_0 + 1 - end - return _accum_0 - end)(), "\n" .. tostring(("~"):rep(80)) .. "\n") - return nomsu - elseif "Action" == _exp_0 then - if inline then - local nomsu = NomsuCode(tree.source) - for i, bit in ipairs(tree) do - if type(bit) == "string" then - if i > 1 then - nomsu:append(" ") - end - nomsu:append(bit) - else - local arg_nomsu = self:tree_to_nomsu(bit, true) - if not (arg_nomsu) then - return nil - end - if not (i == 1) then - nomsu:append(" ") - end - if bit.type == "Action" or bit.type == "Block" then - arg_nomsu:parenthesize() - end - nomsu:append(arg_nomsu) - end - end - return nomsu - else - local nomsu = NomsuCode(tree.source) - local next_space = "" - local line_len, last_colon = 0, nil - for i, bit in ipairs(tree) do - if type(bit) == "string" then - line_len = line_len + #next_space + #bit - nomsu:append(next_space, bit) - next_space = " " - else - local arg_nomsu - if last_colon == i - 1 and bit.type == "Action" then - arg_nomsu = nil - elseif bit.type == "Block" then - arg_nomsu = nil - else - arg_nomsu = self:tree_to_nomsu(bit, true) - end - if arg_nomsu and line_len + #tostring(arg_nomsu) < MAX_LINE then - if bit.type == "Action" then - if can_use_colon and i > 1 then - nomsu:append(match(next_space, "[^ ]*"), ": ", arg_nomsu) - next_space = "\n.." - line_len = 2 - last_colon = i - else - nomsu:append(next_space, "(", arg_nomsu, ")") - line_len = line_len + #next_space + 2 + #tostring(arg_nomsu) - next_space = " " - end - else - nomsu:append(next_space, arg_nomsu) - line_len = line_len + #next_space + #tostring(arg_nomsu) - next_space = " " - end - else - arg_nomsu = self:tree_to_nomsu(bit, nil, true) - if not (nomsu) then - return nil - end - if bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" then - if i == 1 then - arg_nomsu = NomsuCode(bit.source, "(..)\n ", arg_nomsu) - else - arg_nomsu = NomsuCode(bit.source, "\n ", arg_nomsu) - end - end - if last_colon == i - 1 and (bit.type == "Action" or bit.type == "Block") then - next_space = "" - end - nomsu:append(next_space, arg_nomsu) - next_space = "\n.." - line_len = 2 - end - if next_space == " " and #(match(tostring(nomsu), "[^\n]*$")) > MAX_LINE then - next_space = "\n.." - end - end - end - return nomsu - end - elseif "EscapedNomsu" == _exp_0 then - local nomsu = self:tree_to_nomsu(tree[1], true) - if nomsu == nil and not inline then - nomsu = self:tree_to_nomsu(tree[1]) - return nomsu and NomsuCode(tree.source, "\\:\n ", nomsu) - end - return nomsu and NomsuCode(tree.source, "\\(", nomsu, ")") - elseif "Block" == _exp_0 then - if inline then - local nomsu = NomsuCode(tree.source) - for i, line in ipairs(tree) do - if i > 1 then - nomsu:append("; ") - end - local line_nomsu = self:tree_to_nomsu(line, true) - if not (line_nomsu) then - return nil - end - nomsu:append(line_nomsu) - end - return nomsu - end - local nomsu = NomsuCode(tree.source) - for i, line in ipairs(tree) do - line = assert(self:tree_to_nomsu(line, nil, true), "Could not convert line to nomsu") - nomsu:append(line) - if i < #tree then - nomsu:append("\n") - if match(tostring(line), "\n") then - nomsu:append("\n") - end - end - end - return nomsu - elseif "Text" == _exp_0 then - if inline then - local nomsu = NomsuCode(tree.source, '"') - for _index_0 = 1, #tree do - local bit = tree[_index_0] - if type(bit) == 'string' then - nomsu:append((gsub(gsub(gsub(bit, "\\", "\\\\"), "\n", "\\n"), '"', '\\"'))) - else - local interp_nomsu = self:tree_to_nomsu(bit, true) - if interp_nomsu then - if bit.type ~= "Var" and bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" then - interp_nomsu:parenthesize() - end - nomsu:append("\\", interp_nomsu) - else - return nil - end - end - end - nomsu:append('"') - return nomsu - else - local inline_version = self:tree_to_nomsu(tree, true) - if inline_version and #inline_version <= MAX_LINE then - return inline_version - end - local nomsu = NomsuCode(tree.source, '".."\n ') - for i, bit in ipairs(tree) do - if type(bit) == 'string' then - local bit_lines = get_lines:match(bit) - for j, line in ipairs(bit_lines) do - if j > 1 then - nomsu:append("\n ") - end - if #line > 1.25 * MAX_LINE then - local remainder = line - while #remainder > 0 do - local split = find(remainder, " ", MAX_LINE, true) - if split then - local chunk - chunk, remainder = sub(remainder, 1, split), sub(remainder, split + 1, -1) - nomsu:append(chunk) - elseif #remainder > 1.75 * MAX_LINE then - split = math.floor(1.5 * MAX_LINE) - local chunk - chunk, remainder = sub(remainder, 1, split), sub(remainder, split + 1, -1) - nomsu:append(chunk) - else - nomsu:append(remainder) - break - end - if #remainder > 0 then - nomsu:append("\\\n ..") - end - end - else - nomsu:append(line) - end - end - else - local interp_nomsu = self:tree_to_nomsu(bit, true) - if interp_nomsu then - if bit.type ~= "Var" and bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" then - interp_nomsu:parenthesize() - end - nomsu:append("\\", interp_nomsu) - else - interp_nomsu = assert(self:tree_to_nomsu(bit)) - if not (interp_nomsu) then - return nil - end - nomsu:append("\\\n ", interp_nomsu) - if i < #tree then - nomsu:append("\n ..") - end - end - end - end - return nomsu - end - elseif "List" == _exp_0 then - if inline then - local nomsu = NomsuCode(tree.source, "[") - for i, item in ipairs(tree) do - local item_nomsu = self:tree_to_nomsu(item, true) - if not (item_nomsu) then - return nil - end - if i > 1 then - nomsu:append(", ") - end - nomsu:append(item_nomsu) - end - nomsu:append("]") - return nomsu - else - local inline_version = self:tree_to_nomsu(tree, true) - if inline_version and #inline_version <= MAX_LINE then - return inline_version - end - local nomsu = NomsuCode(tree.source, "[..]") - local line = NomsuCode(tree.source, "\n ") - for _index_0 = 1, #tree do - local item = tree[_index_0] - local item_nomsu = self:tree_to_nomsu(item, true) - if item_nomsu and #line + #", " + #item_nomsu <= MAX_LINE then - if #line.bits > 1 then - line:append(", ") - end - line:append(item_nomsu) - else - if not (item_nomsu) then - item_nomsu = self:tree_to_nomsu(item) - if not (item_nomsu) then - return nil - end - end - if #line.bits > 1 then - nomsu:append(line) - line = NomsuCode(line.source, "\n ") - end - line:append(item_nomsu) - end - end - if #line.bits > 1 then - nomsu:append(line) - end - return nomsu - end - elseif "Dict" == _exp_0 then - if inline then - local nomsu = NomsuCode(tree.source, "{") - for i, entry in ipairs(tree) do - local entry_nomsu = self:tree_to_nomsu(entry, true) - if not (entry_nomsu) then - return nil - end - if i > 1 then - nomsu:append(", ") - end - nomsu:append(entry_nomsu) - end - nomsu:append("}") - return nomsu - else - local inline_version = self:tree_to_nomsu(tree, true) - if inline_version then - return inline_version - end - local nomsu = NomsuCode(tree.source, "{..}") - local line = NomsuCode(tree.source, "\n ") - for _index_0 = 1, #tree do - local entry = tree[_index_0] - local entry_nomsu = self:tree_to_nomsu(entry) - if not (entry_nomsu) then - return nil - end - if #line + #tostring(entry_nomsu) <= MAX_LINE then - if #line.bits > 1 then - line:append(", ") - end - line:append(entry_nomsu) - else - if #line.bits > 1 then - nomsu:append(line) - line = NomsuCode(line.source, "\n ") - end - line:append(entry_nomsu) - end - end - if #line.bits > 1 then - nomsu:append(line) - end - return nomsu - end - elseif "DictEntry" == _exp_0 then - local key, value = tree[1], tree[2] - local key_nomsu = self:tree_to_nomsu(key, true) - if not (key_nomsu) then - return nil - end - if key.type == "Action" or key.type == "Block" then - key_nomsu:parenthesize() - end - local value_nomsu - if value then - value_nomsu = self:tree_to_nomsu(value, true) - else - value_nomsu = NomsuCode(tree.source, "") - end - if inline and not value_nomsu then - return nil - end - if not value_nomsu then - if inline then - return nil - end - value_nomsu = self:tree_to_nomsu(value) - if not (value_nomsu) then - return nil - end - end - return NomsuCode(tree.source, key_nomsu, ":", value_nomsu) - elseif "IndexChain" == _exp_0 then - local nomsu = NomsuCode(tree.source) - for i, bit in ipairs(tree) do - if i > 1 then - nomsu:append(".") - end - local bit_nomsu - if bit.type == "Text" and #bit == 1 and type(bit[1]) == 'string' then - if bit[1]:match("[_a-zA-Z][_a-zA-Z0-9]*") then - bit_nomsu = bit[1] - end - end - if not (bit_nomsu) then - bit_nomsu = self:tree_to_nomsu(bit, true) - end - if not (bit_nomsu) then - return nil - end - local _exp_1 = bit.type - if "Action" == _exp_1 or "Block" == _exp_1 or "IndexChain" == _exp_1 then - bit_nomsu:parenthesize() - elseif "Number" == _exp_1 then - if i < #tree then - bit_nomsu:parenthesize() - end - end - nomsu:append(bit_nomsu) - end - return nomsu - elseif "Number" == _exp_0 then - return NomsuCode(tree.source, tostring(tree[1])) - elseif "Var" == _exp_0 then - return NomsuCode(tree.source, "%", tree[1]) - else - return error("Unknown type: " .. tostring(tree.type)) - end - end -end -if arg and debug.getinfo(2).func ~= require then - local parser = re.compile([[ args <- {| (flag ";")* {:inputs: {| ({file} ";")* |} :} {:nomsu_args: {| ("--;" ({[^;]*} ";")*)? |} :} ";"? |} !. - flag <- - {:interactive: ("-i" -> true) :} - / {:optimized: ("-O" -> true) :} - / {:format: ("-f" -> true) :} - / {:syntax: ("-s" -> true) :} - / {:print_file: "-p" ";" {file} :} - / {:compile: ("-c" -> true) :} - / {:verbose: ("-v" -> true) :} - / {:help: (("-h" / "--help") -> true) :} - file <- "-" / [^;]+ - ]], { - ["true"] = function() - return true - end - }) - local args = concat(arg, ";") .. ";" - args = parser:match(args) - if not args or args.help then - print([=[Nomsu Compiler +local usage = [=[Nomsu Compiler Usage: (lua nomsu.lua | moon nomsu.moon) [-i] [-O] [-v] [-c] [-f] [-s] [--help] [-p print_file] file1 file2... [-- nomsu args...] @@ -1181,150 +12,179 @@ OPTIONS -h/--help Print this message. -p Print to the specified file instead of stdout. Input file can be "-" to use stdin. -]=]) - os.exit() +]=] +local lpeg = require('lpeg') +local re = require('re') +local run_safely = require("error_handling") +local NomsuCompiler = require("nomsu_compiler") +local NomsuCode, LuaCode, Source +do + local _obj_0 = require("code_obj") + NomsuCode, LuaCode, Source = _obj_0.NomsuCode, _obj_0.LuaCode, _obj_0.Source +end +local STDIN, STDOUT, STDERR = "/dev/fd/0", "/dev/fd/1", "/dev/fd/2" +if not arg or debug.getinfo(2).func == require then + return NomsuCompiler +end +local parser = re.compile([[ args <- {| (flag ";")* {:inputs: {| ({file} ";")* |} :} {:nomsu_args: {| ("--;" ({[^;]*} ";")*)? |} :} ";"? |} !. + flag <- + {:interactive: ("-i" -> true) :} + / {:optimized: ("-O" -> true) :} + / {:format: ("-f" -> true) :} + / {:syntax: ("-s" -> true) :} + / {:print_file: "-p" ";" {file} :} + / {:compile: ("-c" -> true) :} + / {:verbose: ("-v" -> true) :} + / {:help: (("-h" / "--help") -> true) :} + file <- "-" / [^;]+ +]], { + ["true"] = function() + return true end - local nomsu = NomsuCompiler - nomsu.arg = args.nomsu_args - local run - run = function() - for i, input in ipairs(args.inputs) do - if input == "-" then - args.inputs[i] = STDIN +}) +local args = table.concat(arg, ";") .. ";" +args = parser:match(args) +if not args or args.help then + print(usage) + os.exit() +end +local nomsu = NomsuCompiler +nomsu.arg = args.nomsu_args +local run +run = function() + for i, input in ipairs(args.inputs) do + if input == "-" then + args.inputs[i] = STDIN + end + end + if #args.inputs == 0 and not args.interactive then + args.inputs = { + "core" + } + args.interactive = true + end + local print_file + if args.print_file == "-" then + print_file = io.stdout + elseif args.print_file then + print_file = io.open(args.print_file, 'w') + else + print_file = io.stdout + end + nomsu.skip_precompiled = not args.optimized + if print_file == nil then + nomsu.print = function() end + elseif print_file ~= io.stdout then + nomsu.print = function(...) + local N = select("#", ...) + if N > 0 then + print_file:write(tostring(select(1, ...))) + for i = 2, N do + print_file:write('\t', tostring(select(1, ...))) + end + end + print_file:write('\n') + return print_file:flush() + end + end + local input_files = { } + local to_run = { } + local _list_0 = args.inputs + for _index_0 = 1, #_list_0 do + local input = _list_0[_index_0] + for f in all_files(input) do + input_files[#input_files + 1] = f + to_run[f] = true + end + end + if args.compile or args.verbose then + nomsu.on_compile = function(code, from_file) + if not (to_run[from_file]) then + return + end + if args.verbose then + io.write(tostring(code), "\n") + end + if args.compile and from_file:match("%.nom$") then + local output_filename = from_file:gsub("%.nom$", ".lua") + local output_file = io.open(output_filename, 'w') + output_file:write(tostring(code)) + output_file:flush() + print(("Compiled %-25s -> %s"):format(from_file, output_filename)) + return output_file:close() end end - if #args.inputs == 0 and not args.interactive then - args.inputs = { - "core" - } - args.interactive = true - end - local print_file - if args.print_file == "-" then - print_file = io.stdout - elseif args.print_file then - print_file = io.open(args.print_file, 'w') + end + local parse_errs = { } + for _index_0 = 1, #input_files do + local filename = input_files[_index_0] + if args.syntax then + local file_contents = io.open(filename):read('*a') + local ok, err = pcall(nomsu.parse, nomsu, file_contents, Source(filename, 1, #file_contents)) + if not ok then + table.insert(parse_errs, err) + elseif print_file then + print_file:write("Parse succeeded: " .. tostring(filename) .. "\n") + print_file:flush() + end + elseif args.format then + local file = FILE_CACHE[filename] + if not file then + error("File does not exist: " .. tostring(filename), 0) + end + local tree = nomsu:parse(file, Source(filename, 1, #file)) + local formatted = tostring(nomsu:tree_to_nomsu(tree)) + if print_file then + print_file:write(formatted, "\n") + print_file:flush() + end + elseif filename == STDIN then + local file = io.input():read("*a") + FILE_CACHE.stdin = file + nomsu:run(file, Source('stdin', 1, #file)) else - print_file = io.stdout + nomsu:run_file(filename) end - nomsu.skip_precompiled = not args.optimized - if print_file == nil then - nomsu.print = function() end - elseif print_file ~= io.stdout then - nomsu.print = function(...) - local N = select("#", ...) - if N > 0 then - print_file:write(tostring(select(1, ...))) - for i = 2, N do - print_file:write('\t', tostring(select(1, ...))) + end + if #parse_errs > 0 then + io.stderr:write(table.concat(parse_errs, "\n\n")) + io.stderr:flush() + os.exit(false, true) + elseif args.syntax then + os.exit(true, true) + end + if args.interactive then + for repl_line = 1, math.huge do + io.write(colored.bright(colored.yellow(">> "))) + local buff = { } + while true do + local line = io.read("*L") + if line == "\n" or not line then + if #buff > 0 then + io.write("\027[1A\027[2K") end - end - print_file:write('\n') - return print_file:flush() - end - end - local input_files = { } - local to_run = { } - local _list_0 = args.inputs - for _index_0 = 1, #_list_0 do - local input = _list_0[_index_0] - for f in all_files(input) do - input_files[#input_files + 1] = f - to_run[f] = true - end - end - if args.compile or args.verbose then - nomsu.on_compile = function(code, from_file) - if to_run[from_file] then - if args.verbose then - io.write(tostring(code), "\n") - end - if args.compile and from_file:match("%.nom$") then - local output_filename = from_file:gsub("%.nom$", ".lua") - local output_file = io.open(output_filename, 'w') - output_file:write(tostring(code)) - output_file:flush() - print(("Compiled %-25s -> %s"):format(from_file, output_filename)) - return output_file:close() - end - end - end - else - nomsu.on_compile = nil - end - local parse_errs = { } - for _index_0 = 1, #input_files do - local filename = input_files[_index_0] - if args.syntax then - local file_contents = io.open(filename):read('*a') - local ok, err = pcall(nomsu.parse, nomsu, file_contents, Source(filename, 1, #file_contents)) - if not ok then - insert(parse_errs, err) - elseif print_file then - print_file:write("Parse succeeded: " .. tostring(filename) .. "\n") - print_file:flush() - end - elseif args.format then - local file = FILE_CACHE[filename] - if not file then - error("File does not exist: " .. tostring(filename), 0) - end - local tree = nomsu:parse(file, Source(filename, 1, #file)) - local formatted = tostring(nomsu:tree_to_nomsu(tree)) - if print_file then - print_file:write(formatted, "\n") - print_file:flush() - end - elseif filename == STDIN then - local file = io.input():read("*a") - FILE_CACHE.stdin = file - nomsu:run(file, Source('stdin', 1, #file)) - else - nomsu:run_file(filename) - end - end - if #parse_errs > 0 then - io.stderr:write(concat(parse_errs, "\n\n")) - io.stderr:flush() - os.exit(false, true) - elseif args.syntax then - os.exit(true, true) - end - if args.interactive then - for repl_line = 1, math.huge do - io.write(colored.bright(colored.yellow(">> "))) - local buff = { } - while true do - local line = io.read("*L") - if line == "\n" or not line then - if #buff > 0 then - io.write("\027[1A\027[2K") - end - break - end - line = line:gsub("\t", " ") - insert(buff, line) - io.write(colored.dim(colored.yellow(".. "))) - end - if #buff == 0 then break end - buff = concat(buff) - FILE_CACHE["REPL#" .. repl_line] = buff - local err_hand - err_hand = function(error_message) - return print_err_msg(error_message) - end - local ok, ret = xpcall(nomsu.run, err_hand, nomsu, buff, Source("REPL#" .. repl_line, 1, #buff)) - if ok and ret ~= nil then - print("= " .. repr(ret)) - elseif not ok then - print_err_msg(ret) - end + line = line:gsub("\t", " ") + table.insert(buff, line) + io.write(colored.dim(colored.yellow(".. "))) + end + if #buff == 0 then + break + end + buff = table.concat(buff) + FILE_CACHE["REPL#" .. repl_line] = buff + local err_hand + err_hand = function(error_message) + return print_err_msg(error_message) + end + local ok, ret = xpcall(nomsu.run, err_hand, nomsu, buff, Source("REPL#" .. repl_line, 1, #buff)) + if ok and ret ~= nil then + print("= " .. repr(ret)) + elseif not ok then + print_err_msg(ret) end end end - local run_safely = require("error_handling") - run_safely(run) end -return NomsuCompiler +return run_safely(run) diff --git a/nomsu.moon b/nomsu.moon index 2435920..9dc256e 100755 --- a/nomsu.moon +++ b/nomsu.moon @@ -1,821 +1,6 @@ #!/usr/bin/env moon --- This file contains the source code of the Nomsu compiler. --- Nomsu is a programming language that cross-compiles to Lua. It was designed to be good --- at natural-language-like code that is highly self-modifying and flexible. --- The only dependency is LPEG, which can be installed using "luarocks install lpeg" --- File usage: --- Either, in a lua/moonscript file: --- Nomsu = require "nomsu" --- nomsu = Nomsu() --- nomsu:run(your_nomsu_code) --- Or from the command line: --- lua nomsu.lua your_file.nom -lpeg = require 'lpeg' -re = require 're' -lpeg.setmaxstack 10000 -utils = require 'utils' -{:repr, :stringify, :equivalent} = utils -export colors, colored -colors = require 'consolecolors' -colored = setmetatable({}, {__index:(_,color)-> ((msg)-> colors[color]..tostring(msg or '')..colors.reset)}) -{:insert, :remove, :concat} = table -unpack or= table.unpack -{:match, :sub, :rep, :gsub, :format, :byte, :match, :find} = string -{:NomsuCode, :LuaCode, :Source} = require "code_obj" -AST = require "nomsu_tree" -parse = require("parser") -STDIN, STDOUT, STDERR = "/dev/fd/0", "/dev/fd/1", "/dev/fd/2" --- Mapping from source string (e.g. "@core/metaprogramming.nom[1:100]") to a mapping --- from lua line number to nomsu line number -export SOURCE_MAP -SOURCE_MAP = {} - -string.as_lua_id = (str)-> - argnum = 0 - -- Cut up escape-sequence-like chunks - str = gsub str, "x([0-9A-F][0-9A-F])", "x\0%1" - -- Alphanumeric unchanged, spaces to underscores, and everything else to hex escape sequences - str = gsub str, "%W", (c)-> - if c == ' ' then '_' - elseif c == '%' then - argnum += 1 - tostring(argnum) - else format("x%02X", byte(c)) - return '_'..str - -table.map = (fn)=> [fn(v) for _,v in ipairs(@)] -table.fork = (t, values)-> setmetatable(values or {}, {__index:t}) - --- TODO: --- consider non-linear codegen, rather than doing thunks for things like comprehensions --- Add a ((%x foo %y) where {x:"asdf", y:"fdsa"}) compile-time action for substitution --- Re-implement nomsu-to-lua comment translation? - -export FILE_CACHE --- FILE_CACHE is a map from filename (string) -> string of file contents -FILE_CACHE = setmetatable {}, { - __index: (filename)=> - file = io.open(filename) - return nil unless file - contents = file\read("*a") - file\close! - self[filename] = contents - return contents -} - -iterate_single = (item, prev) -> if item == prev then nil else item -all_files = (path)-> - -- Sanitize path - if match(path, "%.nom$") or match(path, "%.lua$") or match(path, "^/dev/fd/[012]$") - return iterate_single, path - -- TODO: improve sanitization - path = gsub(path,"\\","\\\\") - path = gsub(path,"`","") - path = gsub(path,'"','\\"') - path = gsub(path,"$","") - return coroutine.wrap -> - f = io.popen('find -L "'..path..'" -not -path "*/\\.*" -type f -name "*.nom"') - for line in f\lines! - coroutine.yield(line) - success = f\close! - unless success - error("Invalid file path: "..tostring(path)) - -line_counter = re.compile([[ - lines <- {| line (%nl line)* |} - line <- {} (!%nl .)* -]], nl:lpeg.P("\r")^-1 * lpeg.P("\n")) -get_lines = re.compile([[ - lines <- {| line (%nl line)* |} - line <- {[^%nl]*} -]], nl:lpeg.P("\r")^-1 * lpeg.P("\n")) --- Mapping from line number -> character offset -export LINE_STARTS --- LINE_STARTS is a mapping from strings to a table that maps line number to character positions -LINE_STARTS = setmetatable {}, { - __mode:"k" - __index: (k)=> - -- Implicitly convert Lua and Nomsu objects to strings - if type(k) != 'string' - k = tostring(k) - if v = rawget(self, k) - return v - line_starts = line_counter\match(k) - self[k] = line_starts - return line_starts -} -export pos_to_line -pos_to_line = (str, pos)-> - line_starts = LINE_STARTS[str] - -- Binary search for line number of position - lo, hi = 1, #line_starts - while lo <= hi - mid = math.floor((lo+hi)/2) - if line_starts[mid] > pos - hi = mid-1 - else lo = mid+1 - return hi - --- Use + operator for string coercive concatenation (note: "asdf" + 3 == "asdf3") --- Use [] for accessing string characters, or s[{3,4}] for s:sub(3,4) --- Note: This globally affects all strings in this instance of Lua! -do - STRING_METATABLE = getmetatable("") - STRING_METATABLE.__add = (other)=> @ .. stringify(other) - STRING_METATABLE.__index = (i)=> - ret = string[i] - if ret != nil then return ret - if type(i) == 'number' then return sub(@, i, i) - elseif type(i) == 'table' then return sub(@, i[1], i[2]) - --- List and Dict classes to provide basic equality/tostring functionality for the tables --- used in Nomsu. This way, they retain a notion of whether they were originally lists or dicts. -_list_mt = - __eq:equivalent - -- Could consider adding a __newindex to enforce list-ness, but would hurt performance - __tostring: => - "["..concat([repr(b) for b in *@], ", ").."]" -list = (t)-> setmetatable(t, _list_mt) - -_dict_mt = - __eq:equivalent - __tostring: => - "{"..concat(["#{repr(k)}: #{repr(v)}" for k,v in pairs @], ", ").."}" -dict = (t)-> setmetatable(t, _dict_mt) - -MAX_LINE = 80 -- For beautification purposes, try not to make lines much longer than this value -NomsuCompiler = setmetatable({}, {__index: (k)=> if _self = rawget(@, "self") then _self[k] else nil}) -with NomsuCompiler - ._ENV = NomsuCompiler - .nomsu = NomsuCompiler - .parse = (...)=> parse(...) - - -- Discretionary/convenience stuff - to_add = { - repr:repr, stringify:stringify, utils:utils, lpeg:lpeg, re:re, - -- Lua stuff: - :next, :unpack, :setmetatable, :coroutine, :rawequal, :getmetatable, :pcall, - :error, :package, :os, :require, :tonumber, :tostring, :string, :xpcall, :module, - :print, :loadfile, :rawset, :_VERSION, :collectgarbage, :rawget, :rawlen, - :table, :assert, :dofile, :loadstring, :type, :select, :debug, :math, :io, :load, - :pairs, :ipairs, - -- Nomsu types: - :list, :dict, - } - for k,v in pairs(to_add) do NomsuCompiler[k] = v - for k,v in pairs(AST) do NomsuCompiler[k] = v - .LuaCode = LuaCode - .NomsuCode = NomsuCode - .Source = Source - .ALIASES = setmetatable({}, {__mode:"k"}) - .LOADED = {} - .AST = AST - - .compile_error = (tok, err_format_string, ...)=> - file = FILE_CACHE[tok.source.filename] - line_no = pos_to_line(file, tok.source.start) - line_start = LINE_STARTS[file][line_no] - src = colored.dim(file\sub(line_start, tok.source.start-1)) - src ..= colored.underscore colored.bright colored.red(file\sub(tok.source.start, tok.source.stop-1)) - end_of_line = (LINE_STARTS[file][pos_to_line(file, tok.source.stop) + 1] or 0) - 1 - src ..= colored.dim(file\sub(tok.source.stop, end_of_line-1)) - src = ' '..src\gsub('\n', '\n ') - err_msg = err_format_string\format(src, ...) - error("#{tok.source.filename}:#{line_no}: "..err_msg, 0) - - -- This is a bit of a hack, but this code handles arbitrarily complex - -- math expressions like 2*x + 3^2 without having to define a single - -- action for every possibility. - math_expression = re.compile [[ ([+-] " ")* "%" (" " [*/^+-] (" " [+-])* " %")+ !. ]] - add_lua_bits = (lua, code)=> - for bit in *code - if type(bit) == "string" - lua\append bit - else - bit_lua = @compile(bit) - unless bit_lua.is_value - @compile_error bit, - "Cannot use:\n%s\nas a string interpolation value, since it's not an expression." - lua\append bit_lua - return lua - - add_lua_string_bits = (lua, code)=> - line_len = 0 - if code.type != "Text" - lua\append ", ", @compile(code) - return - for bit in *code - bit_lua = if type(bit) == "string" - repr(bit) - else - bit_lua = @compile(bit) - unless bit_lua.is_value - @compile_error bit, - "Cannot use:\n%s\nas a string interpolation value, since it's not an expression." - bit_lua - line_len += #tostring(bit_lua) - if line_len > MAX_LINE - lua\append ",\n " - line_len = 4 - else - lua\append ", " - lua\append bit_lua - - .COMPILE_ACTIONS = setmetatable { - ["# compile math expr #"]: (tree, ...)=> - lua = LuaCode.Value(tree.source) - for i,tok in ipairs tree - if type(tok) == 'string' - lua\append tok - else - tok_lua = @compile(tok) - unless tok_lua.is_value - @compile_error tok, "Non-expression value inside math expression:\n%s" - if tok.type == "Action" - tok_lua\parenthesize! - lua\append tok_lua - if i < #tree - lua\append " " - return lua - - ["Lua %"]: (tree, _code)=> - lua = LuaCode.Value(_code.source, "LuaCode(", repr(tostring _code.source)) - add_lua_string_bits(@, lua, _code) - lua\append ")" - return lua - - ["Lua value %"]: (tree, _code)=> - lua = LuaCode.Value(_code.source, "LuaCode.Value(", repr(tostring _code.source)) - add_lua_string_bits(@, lua, _code) - lua\append ")" - return lua - - ["lua > %"]: (tree, _code)=> - if _code.type != "Text" - return LuaCode tree.source, "nomsu:run_lua(", @compile(_code), ");" - return add_lua_bits(@, LuaCode(tree.source), _code) - - ["= lua %"]: (tree, _code)=> - if _code.type != "Text" - return LuaCode.Value tree.source, "nomsu:run_lua(", @compile(_code), ":as_statements('return '))" - return add_lua_bits(@, LuaCode.Value(tree.source), _code) - - ["use %"]: (tree, _path)=> - unless _path.type == 'Text' and #_path == 1 and type(_path[1]) == 'string' - return LuaCode(tree.source, "nomsu:run_file(#{@compile(_path)});") - path = _path[1] - @run_file(path) - return LuaCode(tree.source, "nomsu:run_file(#{repr path});") - }, { - __index: (stub)=> - if math_expression\match(stub) - return @["# compile math expr #"] - } - - .run = (to_run, source=nil)=> - tree = if AST.is_syntax_tree(to_run) then tree else @parse(to_run, source or to_run.source) - if tree == nil -- Happens if pattern matches, but there are no captures, e.g. an empty string - return nil - if tree.type == "FileChunks" - -- Each chunk's compilation is affected by the code in the previous chunks - -- (typically), so each chunk needs to compile and run before the next one - -- compiles. - ret = nil - all_lua = {} - for chunk in *tree - lua = @compile(chunk)\as_statements! - lua\declare_locals! - lua\prepend "-- File: #{chunk.source or ""}\n" - insert all_lua, tostring(lua) - ret = @run_lua(lua) - if @on_compile - self.on_compile(concat(all_lua, "\n"), (source or to_run.source).filename) - return ret - else - lua = @compile(tree, compile_actions)\as_statements! - lua\declare_locals! - lua\prepend "-- File: #{source or to_run.source or ""}\n" - if @on_compile - self.on_compile(lua, (source or to_run.source).filename) - return @run_lua(lua) - - _running_files = {} -- For detecting circular imports - .run_file = (filename)=> - if @LOADED[filename] - return @LOADED[filename] - ret = nil - for filename in all_files(filename) - if ret = @LOADED[filename] - continue - - -- Check for circular import - for i,running in ipairs _running_files - if running == filename - loop = [_running_files[j] for j=i,#_running_files] - insert loop, filename - error("Circular import, this loops forever: #{concat loop, " -> "}...") - - insert _running_files, filename - if match(filename, "%.lua$") - file = assert(FILE_CACHE[filename], "Could not find file: #{filename}") - ret = @run_lua file, Source(filename, 1, #file) - elseif match(filename, "%.nom$") or match(filename, "^/dev/fd/[012]$") - ran_lua = if not @skip_precompiled -- Look for precompiled version - lua_filename = gsub(filename, "%.nom$", ".lua") - if file = FILE_CACHE[lua_filename] - ret = @run_lua file, Source(lua_filename, 1, #file) - true - unless ran_lua - file = file or FILE_CACHE[filename] - if not file - error("File does not exist: #{filename}", 0) - ret = @run file, Source(filename,1,#file) - else - error("Invalid filetype for #{filename}", 0) - @LOADED[filename] = ret or true - remove _running_files - - @LOADED[filename] = ret or true - return ret - - .run_lua = (lua, source=nil)=> - lua_string = tostring(lua) - run_lua_fn, err = load(lua_string, nil and tostring(source or lua.source), "t", self) - if not run_lua_fn - line_numbered_lua = concat( - [format("%3d|%s",i,line) for i, line in ipairs get_lines\match(lua_string)], - "\n") - error("Failed to compile generated code:\n#{colored.bright colored.blue colored.onblack line_numbered_lua}\n\n#{err}", 0) - source_key = tostring(source or lua.source) - unless SOURCE_MAP[source_key] - map = {} - offset = 1 - source or= lua.source - nomsu_str = tostring(FILE_CACHE[source.filename]\sub(source.start, source.stop)) - lua_line = 1 - nomsu_line = pos_to_line(nomsu_str, source.start) - fn = (s)-> - if type(s) == 'string' - for nl in s\gmatch("\n") - map[lua_line] or= nomsu_line - lua_line += 1 - else - old_line = nomsu_line - if s.source - nomsu_line = pos_to_line(nomsu_str, s.source.start) - for b in *s.bits do fn(b) - fn(lua) - map[lua_line] or= nomsu_line - map[0] = 0 - -- Mapping from lua line number to nomsu line numbers - SOURCE_MAP[source_key] = map - - return run_lua_fn! - - .compile = (tree)=> - switch tree.type - when "Action" - stub = tree.stub - if compile_action = @COMPILE_ACTIONS[stub] - args = [arg for arg in *tree when type(arg) != "string"] - -- Force Lua to avoid tail call optimization for debugging purposes - -- TODO: use tail call? - ret = compile_action(@, tree, unpack(args)) - if not ret - @compile_error tree, - "Compile-time action:\n%s\nfailed to produce any Lua" - return ret - - lua = LuaCode.Value(tree.source, "A",string.as_lua_id(stub),"(") - args = {} - for i, tok in ipairs tree - if type(tok) == "string" then continue - arg_lua = @compile(tok) - unless arg_lua.is_value - @compile_error tok, - "Cannot use:\n%s\nas an argument to %s, since it's not an expression, it produces: %s", - stub, repr arg_lua - insert args, arg_lua - lua\concat_append args, ", " - lua\append ")" - return lua - - when "EscapedNomsu" - make_tree = (t)-> - unless AST.is_syntax_tree(t) - return repr(t) - bits = [make_tree(bit) for bit in *t] - return t.type.."("..repr(tostring t.source)..", "..table.concat(bits, ", ")..")" - LuaCode.Value tree.source, make_tree(tree[1]) - - when "Block" - lua = LuaCode(tree.source) - lua\concat_append([@compile(line)\as_statements! for line in *tree], "\n") - return lua - - when "Text" - lua = LuaCode.Value(tree.source) - string_buffer = "" - for i, bit in ipairs tree - if type(bit) == "string" - string_buffer ..= bit - continue - if string_buffer ~= "" - if #lua.bits > 0 then lua\append ".." - lua\append repr(string_buffer) - string_buffer = "" - bit_lua = @compile(bit) - unless bit_lua.is_value - src = ' '..gsub(tostring(@tree_to_nomsu(bit)), '\n','\n ') - line = "#{bit.source.filename}:#{pos_to_line(FILE_CACHE[bit.source.filename], bit.source.start)}" - @compile_error bit, - "Cannot use:\n%s\nas a string interpolation value, since it's not an expression." - if #lua.bits > 0 then lua\append ".." - if bit.type != "Text" - bit_lua = LuaCode.Value(bit.source, "stringify(",bit_lua,")") - lua\append bit_lua - - if string_buffer ~= "" or #lua.bits == 0 - if #lua.bits > 0 then lua\append ".." - lua\append repr(string_buffer) - - if #lua.bits > 1 - lua\parenthesize! - return lua - - when "List" - lua = LuaCode.Value tree.source, "list{" - items = {} - for i, item in ipairs tree - item_lua = @compile(item) - unless item_lua.is_value - @compile_error item, - "Cannot use:\n%s\nas a list item, since it's not an expression." - items[i] = item_lua - lua\concat_append(items, ", ", ",\n ") - lua\append "}" - return lua - - when "Dict" - lua = LuaCode.Value tree.source, "dict{" - lua\concat_append([@compile(e) for e in *tree], ", ", ",\n ") - lua\append "}" - return lua - - when "DictEntry" - key, value = tree[1], tree[2] - key_lua = @compile(key) - unless key_lua.is_value - @compile_error tree[1], - "Cannot use:\n%s\nas a dict key, since it's not an expression." - value_lua = value and @compile(value) or LuaCode.Value(key.source, "true") - unless value_lua.is_value - @compile_error tree[2], - "Cannot use:\n%s\nas a dict value, since it's not an expression." - -- TODO: support arbitrary words here, like operators and unicode - key_str = match(tostring(key_lua), [=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=]) - return if key_str - LuaCode tree.source, key_str,"=",value_lua - elseif sub(tostring(key_lua),1,1) == "[" - -- NOTE: this *must* use a space after the [ to avoid freaking out - -- Lua's parser if the inner expression is a long string. Lua - -- parses x[[[y]]] as x("[y]"), not as x["y"] - LuaCode tree.source, "[ ",key_lua,"]=",value_lua - else - LuaCode tree.source, "[",key_lua,"]=",value_lua - - when "IndexChain" - lua = @compile(tree[1]) - unless lua.is_value - @compile_error tree[1], - "Cannot index:\n%s\nsince it's not an expression." - first_char = sub(tostring(lua),1,1) - if first_char == "{" or first_char == '"' or first_char == "[" - lua\parenthesize! - - for i=2,#tree - key = tree[i] - key_lua = @compile(key) - unless key_lua.is_value - @compile_error key, - "Cannot use:\n%s\nas an index, since it's not an expression." - key_lua_str = tostring(key_lua) - if lua_id = match(key_lua_str, "^['\"]([a-zA-Z_][a-zA-Z0-9_]*)['\"]$") - lua\append ".#{lua_id}" - elseif sub(key_lua_str,1,1) == '[' - -- NOTE: this *must* use a space after the [ to avoid freaking out - -- Lua's parser if the inner expression is a long string. Lua - -- parses x[[[y]]] as x("[y]"), not as x["y"] - lua\append "[ ",key_lua," ]" - else - lua\append "[",key_lua,"]" - return lua - - when "Number" - LuaCode.Value(tree.source, tostring(tree[1])) - - when "Var" - LuaCode.Value(tree.source, string.as_lua_id(tree[1])) - - when "FileChunks" - error("Cannot convert FileChunks to a single block of lua, since each chunk's ".. - "compilation depends on the earlier chunks") - - else - error("Unknown type: #{tree.type}") - - .tree_to_nomsu = (tree, inline=false, can_use_colon=false)=> - switch tree.type - when "FileChunks" - return nil if inline - nomsu = NomsuCode(tree.source) - nomsu\concat_append [@tree_to_nomsu(c) for c in *tree], "\n#{("~")\rep(80)}\n" - return nomsu - - when "Action" - if inline - nomsu = NomsuCode(tree.source) - for i,bit in ipairs tree - if type(bit) == "string" - if i > 1 - nomsu\append " " - nomsu\append bit - else - arg_nomsu = @tree_to_nomsu(bit,true) - return nil unless arg_nomsu - unless i == 1 - nomsu\append " " - if bit.type == "Action" or bit.type == "Block" - arg_nomsu\parenthesize! - nomsu\append arg_nomsu - return nomsu - else - nomsu = NomsuCode(tree.source) - next_space = "" - line_len, last_colon = 0, nil - for i,bit in ipairs tree - if type(bit) == "string" - line_len += #next_space + #bit - nomsu\append next_space, bit - next_space = " " - else - arg_nomsu = if last_colon == i-1 and bit.type == "Action" then nil - elseif bit.type == "Block" then nil - else @tree_to_nomsu(bit,true) - - if arg_nomsu and line_len + #tostring(arg_nomsu) < MAX_LINE - if bit.type == "Action" - if can_use_colon and i > 1 - nomsu\append match(next_space,"[^ ]*"), ": ", arg_nomsu - next_space = "\n.." - line_len = 2 - last_colon = i - else - nomsu\append next_space, "(", arg_nomsu, ")" - line_len += #next_space + 2 + #tostring(arg_nomsu) - next_space = " " - else - nomsu\append next_space, arg_nomsu - line_len += #next_space + #tostring(arg_nomsu) - next_space = " " - else - arg_nomsu = @tree_to_nomsu(bit, nil, true) - return nil unless nomsu - -- These types carry their own indentation - if bit.type != "List" and bit.type != "Dict" and bit.type != "Text" - if i == 1 - arg_nomsu = NomsuCode(bit.source, "(..)\n ", arg_nomsu) - else - arg_nomsu = NomsuCode(bit.source, "\n ", arg_nomsu) - - if last_colon == i-1 and (bit.type == "Action" or bit.type == "Block") - next_space = "" - nomsu\append next_space, arg_nomsu - next_space = "\n.." - line_len = 2 - - if next_space == " " and #(match(tostring(nomsu),"[^\n]*$")) > MAX_LINE - next_space = "\n.." - return nomsu - - when "EscapedNomsu" - nomsu = @tree_to_nomsu(tree[1], true) - if nomsu == nil and not inline - nomsu = @tree_to_nomsu(tree[1]) - return nomsu and NomsuCode tree.source, "\\:\n ", nomsu - return nomsu and NomsuCode tree.source, "\\(", nomsu, ")" - - when "Block" - if inline - nomsu = NomsuCode(tree.source) - for i,line in ipairs tree - if i > 1 - nomsu\append "; " - line_nomsu = @tree_to_nomsu(line,true) - return nil unless line_nomsu - nomsu\append line_nomsu - return nomsu - nomsu = NomsuCode(tree.source) - for i, line in ipairs tree - line = assert(@tree_to_nomsu(line, nil, true), "Could not convert line to nomsu") - nomsu\append line - if i < #tree - nomsu\append "\n" - if match(tostring(line), "\n") - nomsu\append "\n" - return nomsu - - when "Text" - if inline - nomsu = NomsuCode(tree.source, '"') - for bit in *tree - if type(bit) == 'string' - -- TODO: unescape better? - nomsu\append (gsub(gsub(gsub(bit,"\\","\\\\"),"\n","\\n"),'"','\\"')) - else - interp_nomsu = @tree_to_nomsu(bit, true) - if interp_nomsu - if bit.type != "Var" and bit.type != "List" and bit.type != "Dict" and bit.type != "Text" - interp_nomsu\parenthesize! - nomsu\append "\\", interp_nomsu - else return nil - nomsu\append '"' - return nomsu - else - inline_version = @tree_to_nomsu(tree, true) - if inline_version and #inline_version <= MAX_LINE - return inline_version - nomsu = NomsuCode(tree.source, '".."\n ') - for i, bit in ipairs tree - if type(bit) == 'string' - bit_lines = get_lines\match(bit) - for j, line in ipairs bit_lines - if j > 1 then nomsu\append "\n " - if #line > 1.25*MAX_LINE - remainder = line - while #remainder > 0 - split = find(remainder, " ", MAX_LINE, true) - if split - chunk, remainder = sub(remainder, 1, split), sub(remainder, split+1, -1) - nomsu\append chunk - elseif #remainder > 1.75*MAX_LINE - split = math.floor(1.5*MAX_LINE) - chunk, remainder = sub(remainder, 1, split), sub(remainder, split+1, -1) - nomsu\append chunk - else - nomsu\append remainder - break - if #remainder > 0 then nomsu\append "\\\n .." - else - nomsu\append line - else - interp_nomsu = @tree_to_nomsu(bit, true) - if interp_nomsu - if bit.type != "Var" and bit.type != "List" and bit.type != "Dict" and bit.type != "Text" - interp_nomsu\parenthesize! - nomsu\append "\\", interp_nomsu - else - interp_nomsu = assert(@tree_to_nomsu(bit)) - return nil unless interp_nomsu - nomsu\append "\\\n ", interp_nomsu - if i < #tree - nomsu\append "\n .." - return nomsu - - when "List" - if inline - nomsu = NomsuCode(tree.source, "[") - for i, item in ipairs tree - item_nomsu = @tree_to_nomsu(item, true) - return nil unless item_nomsu - if i > 1 - nomsu\append ", " - nomsu\append item_nomsu - nomsu\append "]" - return nomsu - else - inline_version = @tree_to_nomsu(tree, true) - if inline_version and #inline_version <= MAX_LINE - return inline_version - nomsu = NomsuCode(tree.source, "[..]") - line = NomsuCode(tree.source, "\n ") - for item in *tree - item_nomsu = @tree_to_nomsu(item, true) - if item_nomsu and #line + #", " + #item_nomsu <= MAX_LINE - if #line.bits > 1 - line\append ", " - line\append item_nomsu - else - unless item_nomsu - item_nomsu = @tree_to_nomsu(item) - return nil unless item_nomsu - if #line.bits > 1 - nomsu\append line - line = NomsuCode(line.source, "\n ") - line\append item_nomsu - if #line.bits > 1 - nomsu\append line - return nomsu - - when "Dict" - if inline - nomsu = NomsuCode(tree.source, "{") - for i, entry in ipairs tree - entry_nomsu = @tree_to_nomsu(entry, true) - return nil unless entry_nomsu - if i > 1 - nomsu\append ", " - nomsu\append entry_nomsu - nomsu\append "}" - return nomsu - else - inline_version = @tree_to_nomsu(tree, true) - if inline_version then return inline_version - nomsu = NomsuCode(tree.source, "{..}") - line = NomsuCode(tree.source, "\n ") - for entry in *tree - entry_nomsu = @tree_to_nomsu(entry) - return nil unless entry_nomsu - if #line + #tostring(entry_nomsu) <= MAX_LINE - if #line.bits > 1 - line\append ", " - line\append entry_nomsu - else - if #line.bits > 1 - nomsu\append line - line = NomsuCode(line.source, "\n ") - line\append entry_nomsu - if #line.bits > 1 - nomsu\append line - return nomsu - - when "DictEntry" - key, value = tree[1], tree[2] - key_nomsu = @tree_to_nomsu(key, true) - return nil unless key_nomsu - if key.type == "Action" or key.type == "Block" - key_nomsu\parenthesize! - value_nomsu = if value - @tree_to_nomsu(value, true) - else NomsuCode(tree.source, "") - if inline and not value_nomsu then return nil - if not value_nomsu - return nil if inline - value_nomsu = @tree_to_nomsu(value) - return nil unless value_nomsu - return NomsuCode tree.source, key_nomsu, ":", value_nomsu - - when "IndexChain" - nomsu = NomsuCode(tree.source) - for i, bit in ipairs tree - if i > 1 - nomsu\append "." - local bit_nomsu - if bit.type == "Text" and #bit == 1 and type(bit[1]) == 'string' - -- TODO: support arbitrary words here, including operators and unicode - if bit[1]\match("[_a-zA-Z][_a-zA-Z0-9]*") - bit_nomsu = bit[1] - unless bit_nomsu then bit_nomsu = @tree_to_nomsu(bit, true) - return nil unless bit_nomsu - switch bit.type - when "Action", "Block", "IndexChain" - bit_nomsu\parenthesize! - when "Number" - if i < #tree - bit_nomsu\parenthesize! - nomsu\append bit_nomsu - return nomsu - - when "Number" - return NomsuCode(tree.source, tostring(tree[1])) - - when "Var" - return NomsuCode(tree.source, "%", tree[1]) - - else - error("Unknown type: #{tree.type}") - - --- Command line interface: --- Only run this code if this file was run directly with command line arguments, and not require()'d: -if arg and debug.getinfo(2).func != require - parser = re.compile([[ - args <- {| (flag ";")* {:inputs: {| ({file} ";")* |} :} {:nomsu_args: {| ("--;" ({[^;]*} ";")*)? |} :} ";"? |} !. - flag <- - {:interactive: ("-i" -> true) :} - / {:optimized: ("-O" -> true) :} - / {:format: ("-f" -> true) :} - / {:syntax: ("-s" -> true) :} - / {:print_file: "-p" ";" {file} :} - / {:compile: ("-c" -> true) :} - / {:verbose: ("-v" -> true) :} - / {:help: (("-h" / "--help") -> true) :} - file <- "-" / [^;]+ - ]], {true: -> true}) - args = concat(arg, ";")..";" - args = parser\match(args) - if not args or args.help - print [=[ +-- This file contains the command-line Nomsu runner. +usage = [=[ Nomsu Compiler Usage: (lua nomsu.lua | moon nomsu.moon) [-i] [-O] [-v] [-c] [-f] [-s] [--help] [-p print_file] file1 file2... [-- nomsu args...] @@ -831,121 +16,145 @@ OPTIONS -p Print to the specified file instead of stdout. Input file can be "-" to use stdin. ]=] - os.exit! - nomsu = NomsuCompiler - nomsu.arg = args.nomsu_args - - run = -> - - for i,input in ipairs args.inputs - if input == "-" then args.inputs[i] = STDIN +lpeg = require 'lpeg' +re = require 're' +run_safely = require "error_handling" +NomsuCompiler = require "nomsu_compiler" +{:NomsuCode, :LuaCode, :Source} = require "code_obj" +STDIN, STDOUT, STDERR = "/dev/fd/0", "/dev/fd/1", "/dev/fd/2" - if #args.inputs == 0 and not args.interactive - args.inputs = {"core"} - args.interactive = true +-- If this file was reached via require(), then just return the Nomsu compiler +if not arg or debug.getinfo(2).func == require + return NomsuCompiler - print_file = if args.print_file == "-" then io.stdout - elseif args.print_file then io.open(args.print_file, 'w') - else io.stdout +parser = re.compile([[ + args <- {| (flag ";")* {:inputs: {| ({file} ";")* |} :} {:nomsu_args: {| ("--;" ({[^;]*} ";")*)? |} :} ";"? |} !. + flag <- + {:interactive: ("-i" -> true) :} + / {:optimized: ("-O" -> true) :} + / {:format: ("-f" -> true) :} + / {:syntax: ("-s" -> true) :} + / {:print_file: "-p" ";" {file} :} + / {:compile: ("-c" -> true) :} + / {:verbose: ("-v" -> true) :} + / {:help: (("-h" / "--help") -> true) :} + file <- "-" / [^;]+ +]], {true: -> true}) +args = table.concat(arg, ";")..";" +args = parser\match(args) +if not args or args.help + print usage + os.exit! - nomsu.skip_precompiled = not args.optimized - if print_file == nil - nomsu.print = -> - elseif print_file != io.stdout - nomsu.print = (...)-> - N = select("#",...) - if N > 0 - print_file\write(tostring(select(1,...))) - for i=2,N - print_file\write('\t',tostring(select(1,...))) - print_file\write('\n') +nomsu = NomsuCompiler +nomsu.arg = args.nomsu_args + +run = -> + for i,input in ipairs args.inputs + if input == "-" then args.inputs[i] = STDIN + + if #args.inputs == 0 and not args.interactive + args.inputs = {"core"} + args.interactive = true + + print_file = if args.print_file == "-" then io.stdout + elseif args.print_file then io.open(args.print_file, 'w') + else io.stdout + + nomsu.skip_precompiled = not args.optimized + if print_file == nil + nomsu.print = -> + elseif print_file != io.stdout + nomsu.print = (...)-> + N = select("#",...) + if N > 0 + print_file\write(tostring(select(1,...))) + for i=2,N + print_file\write('\t',tostring(select(1,...))) + print_file\write('\n') + print_file\flush! + + input_files = {} + to_run = {} + for input in *args.inputs + for f in all_files(input) + input_files[#input_files+1] = f + to_run[f] = true + + if args.compile or args.verbose + nomsu.on_compile = (code, from_file)-> + return unless to_run[from_file] + if args.verbose + io.write(tostring(code), "\n") + if args.compile and from_file\match("%.nom$") + output_filename = from_file\gsub("%.nom$", ".lua") + output_file = io.open(output_filename, 'w') + output_file\write(tostring(code)) + output_file\flush! + print ("Compiled %-25s -> %s")\format(from_file, output_filename) + output_file\close! + + parse_errs = {} + for filename in *input_files + if args.syntax + -- Check syntax: + file_contents = io.open(filename)\read('*a') + ok,err = pcall nomsu.parse, nomsu, file_contents, Source(filename, 1, #file_contents) + if not ok + table.insert parse_errs, err + elseif print_file + print_file\write("Parse succeeded: #{filename}\n") print_file\flush! + elseif args.format + -- Auto-format + file = FILE_CACHE[filename] + if not file + error("File does not exist: #{filename}", 0) + tree = nomsu\parse(file, Source(filename,1,#file)) + formatted = tostring(nomsu\tree_to_nomsu(tree)) + if print_file + print_file\write(formatted, "\n") + print_file\flush! + elseif filename == STDIN + file = io.input!\read("*a") + FILE_CACHE.stdin = file + nomsu\run(file, Source('stdin',1,#file)) + else + nomsu\run_file(filename) - input_files = {} - to_run = {} - for input in *args.inputs - for f in all_files(input) - input_files[#input_files+1] = f - to_run[f] = true + if #parse_errs > 0 + io.stderr\write table.concat(parse_errs, "\n\n") + io.stderr\flush! + os.exit(false, true) + elseif args.syntax + os.exit(true, true) - nomsu.on_compile = if args.compile or args.verbose - (code, from_file)-> - if to_run[from_file] - if args.verbose - io.write(tostring(code), "\n") - if args.compile and from_file\match("%.nom$") - output_filename = from_file\gsub("%.nom$", ".lua") - output_file = io.open(output_filename, 'w') - output_file\write(tostring(code)) - output_file\flush! - print ("Compiled %-25s -> %s")\format(from_file, output_filename) - output_file\close! - else nil + if args.interactive + -- REPL + for repl_line=1,math.huge + io.write(colored.bright colored.yellow ">> ") + buff = {} + while true + line = io.read("*L") + if line == "\n" or not line + if #buff > 0 + io.write("\027[1A\027[2K") + break -- Run buffer + line = line\gsub("\t", " ") + table.insert buff, line + io.write(colored.dim colored.yellow ".. ") + if #buff == 0 + break -- Exit + + buff = table.concat(buff) + FILE_CACHE["REPL#"..repl_line] = buff + err_hand = (error_message)-> + print_err_msg error_message + ok, ret = xpcall(nomsu.run, err_hand, nomsu, buff, Source("REPL#"..repl_line, 1, #buff)) + if ok and ret != nil + print "= "..repr(ret) + elseif not ok + print_err_msg ret - parse_errs = {} - for filename in *input_files - if args.syntax - -- Check syntax: - file_contents = io.open(filename)\read('*a') - ok,err = pcall nomsu.parse, nomsu, file_contents, Source(filename, 1, #file_contents) - if not ok - insert parse_errs, err - elseif print_file - print_file\write("Parse succeeded: #{filename}\n") - print_file\flush! - elseif args.format - -- Auto-format - file = FILE_CACHE[filename] - if not file - error("File does not exist: #{filename}", 0) - tree = nomsu\parse(file, Source(filename,1,#file)) - formatted = tostring(nomsu\tree_to_nomsu(tree)) - if print_file - print_file\write(formatted, "\n") - print_file\flush! - elseif filename == STDIN - file = io.input!\read("*a") - FILE_CACHE.stdin = file - nomsu\run(file, Source('stdin',1,#file)) - else - nomsu\run_file(filename) - - if #parse_errs > 0 - io.stderr\write concat(parse_errs, "\n\n") - io.stderr\flush! - os.exit(false, true) - elseif args.syntax - os.exit(true, true) - - if args.interactive - -- REPL - for repl_line=1,math.huge - io.write(colored.bright colored.yellow ">> ") - buff = {} - while true - line = io.read("*L") - if line == "\n" or not line - if #buff > 0 - io.write("\027[1A\027[2K") - break -- Run buffer - line = line\gsub("\t", " ") - insert buff, line - io.write(colored.dim colored.yellow ".. ") - if #buff == 0 - break -- Exit - - buff = concat(buff) - FILE_CACHE["REPL#"..repl_line] = buff - err_hand = (error_message)-> - print_err_msg error_message - ok, ret = xpcall(nomsu.run, err_hand, nomsu, buff, Source("REPL#"..repl_line, 1, #buff)) - if ok and ret != nil - print "= "..repr(ret) - elseif not ok - print_err_msg ret - - run_safely = require "error_handling" - run_safely(run) - -return NomsuCompiler +run_safely(run) diff --git a/nomsu_compiler.lua b/nomsu_compiler.lua new file mode 100644 index 0000000..d093ff5 --- /dev/null +++ b/nomsu_compiler.lua @@ -0,0 +1,1148 @@ +local lpeg = require('lpeg') +local re = require('re') +lpeg.setmaxstack(10000) +local utils = require('utils') +local repr, stringify, equivalent +repr, stringify, equivalent = utils.repr, utils.stringify, utils.equivalent +colors = require('consolecolors') +colored = setmetatable({ }, { + __index = function(_, color) + return (function(msg) + return colors[color] .. tostring(msg or '') .. colors.reset + end) + 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, rep, gsub, format, byte, find +do + local _obj_0 = string + match, sub, rep, gsub, format, byte, match, find = _obj_0.match, _obj_0.sub, _obj_0.rep, _obj_0.gsub, _obj_0.format, _obj_0.byte, _obj_0.match, _obj_0.find +end +local NomsuCode, LuaCode, Source +do + local _obj_0 = require("code_obj") + NomsuCode, LuaCode, Source = _obj_0.NomsuCode, _obj_0.LuaCode, _obj_0.Source +end +local AST = require("nomsu_tree") +local parse = require("parser") +SOURCE_MAP = { } +string.as_lua_id = function(str) + local argnum = 0 + str = gsub(str, "x([0-9A-F][0-9A-F])", "x\0%1") + str = gsub(str, "%W", function(c) + if c == ' ' then + return '_' + elseif c == '%' then + argnum = argnum + 1 + return tostring(argnum) + else + return format("x%02X", byte(c)) + end + end) + return '_' .. str +end +table.map = function(self, fn) + local _accum_0 = { } + local _len_0 = 1 + for _, v in ipairs(self) do + _accum_0[_len_0] = fn(v) + _len_0 = _len_0 + 1 + end + return _accum_0 +end +table.fork = function(t, values) + return setmetatable(values or { }, { + __index = t + }) +end +FILE_CACHE = setmetatable({ }, { + __index = function(self, filename) + local file = io.open(filename) + if not (file) then + return nil + end + local contents = file:read("*a") + file:close() + self[filename] = contents + return contents + end +}) +local iterate_single +iterate_single = function(item, prev) + if item == prev then + return nil + else + return item + end +end +all_files = function(path) + if match(path, "%.nom$") or match(path, "%.lua$") or match(path, "^/dev/fd/[012]$") then + return iterate_single, path + end + path = gsub(path, "\\", "\\\\") + path = gsub(path, "`", "") + path = gsub(path, '"', '\\"') + path = gsub(path, "$", "") + return coroutine.wrap(function() + local f = io.popen('find -L "' .. path .. '" -not -path "*/\\.*" -type f -name "*.nom"') + for line in f:lines() do + coroutine.yield(line) + end + local success = f:close() + if not (success) then + return error("Invalid file path: " .. tostring(path)) + end + end) +end +local line_counter = re.compile([[ lines <- {| line (%nl line)* |} + line <- {} (!%nl .)* +]], { + nl = lpeg.P("\r") ^ -1 * lpeg.P("\n") +}) +local get_lines = re.compile([[ lines <- {| line (%nl line)* |} + line <- {[^%nl]*} +]], { + nl = lpeg.P("\r") ^ -1 * lpeg.P("\n") +}) +LINE_STARTS = setmetatable({ }, { + __mode = "k", + __index = function(self, k) + if type(k) ~= 'string' then + k = tostring(k) + do + local v = rawget(self, k) + if v then + return v + end + end + end + local line_starts = line_counter:match(k) + self[k] = line_starts + return line_starts + end +}) +pos_to_line = function(str, pos) + local line_starts = LINE_STARTS[str] + local lo, hi = 1, #line_starts + while lo <= hi do + local mid = math.floor((lo + hi) / 2) + if line_starts[mid] > pos then + hi = mid - 1 + else + lo = mid + 1 + end + end + return hi +end +do + local STRING_METATABLE = getmetatable("") + STRING_METATABLE.__add = function(self, other) + return self .. stringify(other) + end + STRING_METATABLE.__index = function(self, i) + local ret = string[i] + if ret ~= nil then + return ret + end + if type(i) == 'number' then + return sub(self, i, i) + elseif type(i) == 'table' then + return sub(self, i[1], i[2]) + end + end +end +local _list_mt = { + __eq = equivalent, + __tostring = function(self) + return "[" .. concat((function() + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 1, #self do + local b = self[_index_0] + _accum_0[_len_0] = repr(b) + _len_0 = _len_0 + 1 + end + return _accum_0 + end)(), ", ") .. "]" + end +} +local list +list = function(t) + return setmetatable(t, _list_mt) +end +local _dict_mt = { + __eq = equivalent, + __tostring = function(self) + return "{" .. concat((function() + local _accum_0 = { } + local _len_0 = 1 + for k, v in pairs(self) do + _accum_0[_len_0] = tostring(repr(k)) .. ": " .. tostring(repr(v)) + _len_0 = _len_0 + 1 + end + return _accum_0 + end)(), ", ") .. "}" + end +} +local dict +dict = function(t) + return setmetatable(t, _dict_mt) +end +local MAX_LINE = 80 +local NomsuCompiler = setmetatable({ }, { + __index = function(self, k) + do + local _self = rawget(self, "self") + if _self then + return _self[k] + else + return nil + end + end + end +}) +do + NomsuCompiler._ENV = NomsuCompiler + NomsuCompiler.nomsu = NomsuCompiler + NomsuCompiler.parse = function(self, ...) + return parse(...) + end + local to_add = { + repr = repr, + stringify = stringify, + utils = utils, + lpeg = lpeg, + re = re, + next = next, + unpack = unpack, + setmetatable = setmetatable, + coroutine = coroutine, + rawequal = rawequal, + getmetatable = getmetatable, + pcall = pcall, + error = error, + package = package, + os = os, + require = require, + tonumber = tonumber, + tostring = tostring, + string = string, + xpcall = xpcall, + module = module, + print = print, + loadfile = loadfile, + rawset = rawset, + _VERSION = _VERSION, + collectgarbage = collectgarbage, + rawget = rawget, + rawlen = rawlen, + table = table, + assert = assert, + dofile = dofile, + loadstring = loadstring, + type = type, + select = select, + debug = debug, + math = math, + io = io, + load = load, + pairs = pairs, + ipairs = ipairs, + list = list, + dict = dict + } + for k, v in pairs(to_add) do + NomsuCompiler[k] = v + end + for k, v in pairs(AST) do + NomsuCompiler[k] = v + end + NomsuCompiler.LuaCode = LuaCode + NomsuCompiler.NomsuCode = NomsuCode + NomsuCompiler.Source = Source + NomsuCompiler.ALIASES = setmetatable({ }, { + __mode = "k" + }) + NomsuCompiler.LOADED = { } + NomsuCompiler.AST = AST + NomsuCompiler.compile_error = function(self, tok, err_format_string, ...) + local file = FILE_CACHE[tok.source.filename] + local line_no = pos_to_line(file, tok.source.start) + local line_start = LINE_STARTS[file][line_no] + local src = colored.dim(file:sub(line_start, tok.source.start - 1)) + src = src .. colored.underscore(colored.bright(colored.red(file:sub(tok.source.start, tok.source.stop - 1)))) + local end_of_line = (LINE_STARTS[file][pos_to_line(file, tok.source.stop) + 1] or 0) - 1 + src = src .. colored.dim(file:sub(tok.source.stop, end_of_line - 1)) + src = ' ' .. src:gsub('\n', '\n ') + local err_msg = err_format_string:format(src, ...) + return error(tostring(tok.source.filename) .. ":" .. tostring(line_no) .. ": " .. err_msg, 0) + end + local math_expression = re.compile([[ ([+-] " ")* "%" (" " [*/^+-] (" " [+-])* " %")+ !. ]]) + local add_lua_bits + add_lua_bits = function(self, lua, code) + for _index_0 = 1, #code do + local bit = code[_index_0] + if type(bit) == "string" then + lua:append(bit) + else + local bit_lua = self:compile(bit) + if not (bit_lua.is_value) then + self:compile_error(bit, "Cannot use:\n%s\nas a string interpolation value, since it's not an expression.") + end + lua:append(bit_lua) + end + end + return lua + end + local add_lua_string_bits + add_lua_string_bits = function(self, lua, code) + local line_len = 0 + if code.type ~= "Text" then + lua:append(", ", self:compile(code)) + return + end + for _index_0 = 1, #code do + local bit = code[_index_0] + local bit_lua + if type(bit) == "string" then + bit_lua = repr(bit) + else + bit_lua = self:compile(bit) + if not (bit_lua.is_value) then + self:compile_error(bit, "Cannot use:\n%s\nas a string interpolation value, since it's not an expression.") + end + bit_lua = bit_lua + end + line_len = line_len + #tostring(bit_lua) + if line_len > MAX_LINE then + lua:append(",\n ") + line_len = 4 + else + lua:append(", ") + end + lua:append(bit_lua) + end + end + NomsuCompiler.COMPILE_ACTIONS = setmetatable({ + ["# compile math expr #"] = function(self, tree, ...) + local lua = LuaCode.Value(tree.source) + for i, tok in ipairs(tree) do + if type(tok) == 'string' then + lua:append(tok) + else + local tok_lua = self:compile(tok) + if not (tok_lua.is_value) then + self:compile_error(tok, "Non-expression value inside math expression:\n%s") + end + 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, + ["Lua %"] = function(self, tree, _code) + local lua = LuaCode.Value(_code.source, "LuaCode(", repr(tostring(_code.source))) + add_lua_string_bits(self, lua, _code) + lua:append(")") + return lua + end, + ["Lua value %"] = function(self, tree, _code) + local lua = LuaCode.Value(_code.source, "LuaCode.Value(", repr(tostring(_code.source))) + add_lua_string_bits(self, lua, _code) + lua:append(")") + return lua + end, + ["lua > %"] = function(self, tree, _code) + if _code.type ~= "Text" then + return LuaCode(tree.source, "nomsu:run_lua(", self:compile(_code), ");") + end + return add_lua_bits(self, LuaCode(tree.source), _code) + end, + ["= lua %"] = function(self, tree, _code) + if _code.type ~= "Text" then + return LuaCode.Value(tree.source, "nomsu:run_lua(", self:compile(_code), ":as_statements('return '))") + end + return add_lua_bits(self, LuaCode.Value(tree.source), _code) + end, + ["use %"] = function(self, tree, _path) + if not (_path.type == 'Text' and #_path == 1 and type(_path[1]) == 'string') then + return LuaCode(tree.source, "nomsu:run_file(" .. tostring(self:compile(_path)) .. ");") + end + local path = _path[1] + self:run_file(path) + return LuaCode(tree.source, "nomsu:run_file(" .. tostring(repr(path)) .. ");") + end + }, { + __index = function(self, stub) + if math_expression:match(stub) then + return self["# compile math expr #"] + end + end + }) + NomsuCompiler.run = function(self, to_run, source) + if source == nil then + source = nil + end + local tree + if AST.is_syntax_tree(to_run) then + tree = tree + else + tree = self:parse(to_run, source or to_run.source) + end + if tree == nil then + return nil + end + if tree.type == "FileChunks" then + local ret = nil + local all_lua = { } + for _index_0 = 1, #tree do + local chunk = tree[_index_0] + local lua = self:compile(chunk):as_statements() + lua:declare_locals() + lua:prepend("-- File: " .. tostring(chunk.source or "") .. "\n") + insert(all_lua, tostring(lua)) + ret = self:run_lua(lua) + end + if self.on_compile then + self.on_compile(concat(all_lua, "\n"), (source or to_run.source).filename) + end + return ret + else + local lua = self:compile(tree, compile_actions):as_statements() + lua:declare_locals() + lua:prepend("-- File: " .. tostring(source or to_run.source or "") .. "\n") + if self.on_compile then + self.on_compile(lua, (source or to_run.source).filename) + end + return self:run_lua(lua) + end + end + local _running_files = { } + NomsuCompiler.run_file = function(self, filename) + if self.LOADED[filename] then + return self.LOADED[filename] + end + local ret = nil + for filename in all_files(filename) do + local _continue_0 = false + repeat + do + ret = self.LOADED[filename] + if ret then + _continue_0 = true + break + end + end + for i, running in ipairs(_running_files) do + if running == filename then + local loop + do + local _accum_0 = { } + local _len_0 = 1 + for j = i, #_running_files do + _accum_0[_len_0] = _running_files[j] + _len_0 = _len_0 + 1 + end + loop = _accum_0 + end + insert(loop, filename) + error("Circular import, this loops forever: " .. tostring(concat(loop, " -> ")) .. "...") + end + end + insert(_running_files, filename) + if match(filename, "%.lua$") then + local file = assert(FILE_CACHE[filename], "Could not find file: " .. tostring(filename)) + ret = self:run_lua(file, Source(filename, 1, #file)) + elseif match(filename, "%.nom$") or match(filename, "^/dev/fd/[012]$") then + local ran_lua + if not self.skip_precompiled then + local lua_filename = gsub(filename, "%.nom$", ".lua") + do + local file = FILE_CACHE[lua_filename] + if file then + ret = self:run_lua(file, Source(lua_filename, 1, #file)) + ran_lua = true + end + end + end + if not (ran_lua) then + local file = file or FILE_CACHE[filename] + if not file then + error("File does not exist: " .. tostring(filename), 0) + end + ret = self:run(file, Source(filename, 1, #file)) + end + else + error("Invalid filetype for " .. tostring(filename), 0) + end + self.LOADED[filename] = ret or true + remove(_running_files) + _continue_0 = true + until true + if not _continue_0 then + break + end + end + self.LOADED[filename] = ret or true + return ret + end + NomsuCompiler.run_lua = function(self, lua, source) + if source == nil then + source = nil + end + local lua_string = tostring(lua) + local run_lua_fn, err = load(lua_string, nil and tostring(source or lua.source), "t", self) + if not run_lua_fn then + local line_numbered_lua = concat((function() + local _accum_0 = { } + local _len_0 = 1 + for i, line in ipairs(get_lines:match(lua_string)) do + _accum_0[_len_0] = format("%3d|%s", i, line) + _len_0 = _len_0 + 1 + end + return _accum_0 + end)(), "\n") + error("Failed to compile generated code:\n" .. tostring(colored.bright(colored.blue(colored.onblack(line_numbered_lua)))) .. "\n\n" .. tostring(err), 0) + end + local source_key = tostring(source or lua.source) + if not (SOURCE_MAP[source_key]) then + local map = { } + local offset = 1 + source = source or lua.source + local nomsu_str = tostring(FILE_CACHE[source.filename]:sub(source.start, source.stop)) + local lua_line = 1 + local nomsu_line = pos_to_line(nomsu_str, source.start) + local fn + fn = function(s) + if type(s) == 'string' then + for nl in s:gmatch("\n") do + map[lua_line] = map[lua_line] or nomsu_line + lua_line = lua_line + 1 + end + else + local old_line = nomsu_line + if s.source then + nomsu_line = pos_to_line(nomsu_str, s.source.start) + end + local _list_0 = s.bits + for _index_0 = 1, #_list_0 do + local b = _list_0[_index_0] + fn(b) + end + end + end + fn(lua) + map[lua_line] = map[lua_line] or nomsu_line + map[0] = 0 + SOURCE_MAP[source_key] = map + end + return run_lua_fn() + end + NomsuCompiler.compile = function(self, tree) + local _exp_0 = tree.type + if "Action" == _exp_0 then + local stub = tree.stub + do + local compile_action = self.COMPILE_ACTIONS[stub] + 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 + end + local ret = compile_action(self, tree, unpack(args)) + if not ret then + self:compile_error(tree, "Compile-time action:\n%s\nfailed to produce any Lua") + end + return ret + end + end + local lua = LuaCode.Value(tree.source, "A", string.as_lua_id(stub), "(") + 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 = self:compile(tok) + if not (arg_lua.is_value) then + self:compile_error(tok, "Cannot use:\n%s\nas an argument to %s, since it's not an expression, it produces: %s", stub, repr(arg_lua)) + 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 make_tree + make_tree = function(t) + if not (AST.is_syntax_tree(t)) then + return repr(t) + end + local bits + do + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 1, #t do + local bit = t[_index_0] + _accum_0[_len_0] = make_tree(bit) + _len_0 = _len_0 + 1 + end + bits = _accum_0 + end + return t.type .. "(" .. repr(tostring(t.source)) .. ", " .. table.concat(bits, ", ") .. ")" + end + return LuaCode.Value(tree.source, make_tree(tree[1])) + elseif "Block" == _exp_0 then + local lua = LuaCode(tree.source) + lua:concat_append((function() + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 1, #tree do + local line = tree[_index_0] + _accum_0[_len_0] = self:compile(line):as_statements() + _len_0 = _len_0 + 1 + end + return _accum_0 + end)(), "\n") + return lua + elseif "Text" == _exp_0 then + local lua = LuaCode.Value(tree.source) + 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 + if #lua.bits > 0 then + lua:append("..") + end + lua:append(repr(string_buffer)) + string_buffer = "" + end + local bit_lua = self:compile(bit) + if not (bit_lua.is_value) then + local src = ' ' .. gsub(tostring(self:tree_to_nomsu(bit)), '\n', '\n ') + local line = tostring(bit.source.filename) .. ":" .. tostring(pos_to_line(FILE_CACHE[bit.source.filename], bit.source.start)) + self:compile_error(bit, "Cannot use:\n%s\nas a string interpolation value, since it's not an expression.") + end + if #lua.bits > 0 then + lua:append("..") + end + if bit.type ~= "Text" then + bit_lua = LuaCode.Value(bit.source, "stringify(", bit_lua, ")") + end + lua:append(bit_lua) + _continue_0 = true + until true + if not _continue_0 then + break + end + end + if string_buffer ~= "" or #lua.bits == 0 then + if #lua.bits > 0 then + lua:append("..") + end + lua:append(repr(string_buffer)) + end + if #lua.bits > 1 then + lua:parenthesize() + end + return lua + elseif "List" == _exp_0 then + local lua = LuaCode.Value(tree.source, "list{") + local items = { } + for i, item in ipairs(tree) do + local item_lua = self:compile(item) + if not (item_lua.is_value) then + self:compile_error(item, "Cannot use:\n%s\nas a list item, since it's not an expression.") + end + items[i] = item_lua + end + lua:concat_append(items, ", ", ",\n ") + lua:append("}") + return lua + elseif "Dict" == _exp_0 then + local lua = LuaCode.Value(tree.source, "dict{") + lua:concat_append((function() + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 1, #tree do + local e = tree[_index_0] + _accum_0[_len_0] = self:compile(e) + _len_0 = _len_0 + 1 + end + return _accum_0 + end)(), ", ", ",\n ") + lua:append("}") + return lua + elseif "DictEntry" == _exp_0 then + local key, value = tree[1], tree[2] + local key_lua = self:compile(key) + if not (key_lua.is_value) then + self:compile_error(tree[1], "Cannot use:\n%s\nas a dict key, since it's not an expression.") + end + local value_lua = value and self:compile(value) or LuaCode.Value(key.source, "true") + if not (value_lua.is_value) then + self:compile_error(tree[2], "Cannot use:\n%s\nas a dict value, since it's not an expression.") + end + local key_str = match(tostring(key_lua), [=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=]) + if key_str then + return LuaCode(tree.source, key_str, "=", value_lua) + elseif sub(tostring(key_lua), 1, 1) == "[" then + return LuaCode(tree.source, "[ ", key_lua, "]=", value_lua) + else + return LuaCode(tree.source, "[", key_lua, "]=", value_lua) + end + elseif "IndexChain" == _exp_0 then + local lua = self:compile(tree[1]) + if not (lua.is_value) then + self:compile_error(tree[1], "Cannot index:\n%s\nsince it's not an expression.") + end + local first_char = sub(tostring(lua), 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 = self:compile(key) + if not (key_lua.is_value) then + self:compile_error(key, "Cannot use:\n%s\nas an index, since it's not an expression.") + end + local key_lua_str = tostring(key_lua) + do + local lua_id = match(key_lua_str, "^['\"]([a-zA-Z_][a-zA-Z0-9_]*)['\"]$") + if 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 + end + return lua + elseif "Number" == _exp_0 then + return LuaCode.Value(tree.source, tostring(tree[1])) + elseif "Var" == _exp_0 then + return LuaCode.Value(tree.source, string.as_lua_id(tree[1])) + elseif "FileChunks" == _exp_0 then + return error("Cannot convert FileChunks to a single block of lua, since each chunk's " .. "compilation depends on the earlier chunks") + else + return error("Unknown type: " .. tostring(tree.type)) + end + end + NomsuCompiler.tree_to_nomsu = function(self, tree, inline, can_use_colon) + if inline == nil then + inline = false + end + if can_use_colon == nil then + can_use_colon = false + end + local _exp_0 = tree.type + if "FileChunks" == _exp_0 then + if inline then + return nil + end + local nomsu = NomsuCode(tree.source) + nomsu:concat_append((function() + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 1, #tree do + local c = tree[_index_0] + _accum_0[_len_0] = self:tree_to_nomsu(c) + _len_0 = _len_0 + 1 + end + return _accum_0 + end)(), "\n" .. tostring(("~"):rep(80)) .. "\n") + return nomsu + elseif "Action" == _exp_0 then + if inline then + local nomsu = NomsuCode(tree.source) + for i, bit in ipairs(tree) do + if type(bit) == "string" then + if i > 1 then + nomsu:append(" ") + end + nomsu:append(bit) + else + local arg_nomsu = self:tree_to_nomsu(bit, true) + if not (arg_nomsu) then + return nil + end + if not (i == 1) then + nomsu:append(" ") + end + if bit.type == "Action" or bit.type == "Block" then + arg_nomsu:parenthesize() + end + nomsu:append(arg_nomsu) + end + end + return nomsu + else + local nomsu = NomsuCode(tree.source) + local next_space = "" + local line_len, last_colon = 0, nil + for i, bit in ipairs(tree) do + if type(bit) == "string" then + line_len = line_len + #next_space + #bit + nomsu:append(next_space, bit) + next_space = " " + else + local arg_nomsu + if last_colon == i - 1 and bit.type == "Action" then + arg_nomsu = nil + elseif bit.type == "Block" then + arg_nomsu = nil + else + arg_nomsu = self:tree_to_nomsu(bit, true) + end + if arg_nomsu and line_len + #tostring(arg_nomsu) < MAX_LINE then + if bit.type == "Action" then + if can_use_colon and i > 1 then + nomsu:append(match(next_space, "[^ ]*"), ": ", arg_nomsu) + next_space = "\n.." + line_len = 2 + last_colon = i + else + nomsu:append(next_space, "(", arg_nomsu, ")") + line_len = line_len + #next_space + 2 + #tostring(arg_nomsu) + next_space = " " + end + else + nomsu:append(next_space, arg_nomsu) + line_len = line_len + #next_space + #tostring(arg_nomsu) + next_space = " " + end + else + arg_nomsu = self:tree_to_nomsu(bit, nil, true) + if not (nomsu) then + return nil + end + if bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" then + if i == 1 then + arg_nomsu = NomsuCode(bit.source, "(..)\n ", arg_nomsu) + else + arg_nomsu = NomsuCode(bit.source, "\n ", arg_nomsu) + end + end + if last_colon == i - 1 and (bit.type == "Action" or bit.type == "Block") then + next_space = "" + end + nomsu:append(next_space, arg_nomsu) + next_space = "\n.." + line_len = 2 + end + if next_space == " " and #(match(tostring(nomsu), "[^\n]*$")) > MAX_LINE then + next_space = "\n.." + end + end + end + return nomsu + end + elseif "EscapedNomsu" == _exp_0 then + local nomsu = self:tree_to_nomsu(tree[1], true) + if nomsu == nil and not inline then + nomsu = self:tree_to_nomsu(tree[1]) + return nomsu and NomsuCode(tree.source, "\\:\n ", nomsu) + end + return nomsu and NomsuCode(tree.source, "\\(", nomsu, ")") + elseif "Block" == _exp_0 then + if inline then + local nomsu = NomsuCode(tree.source) + for i, line in ipairs(tree) do + if i > 1 then + nomsu:append("; ") + end + local line_nomsu = self:tree_to_nomsu(line, true) + if not (line_nomsu) then + return nil + end + nomsu:append(line_nomsu) + end + return nomsu + end + local nomsu = NomsuCode(tree.source) + for i, line in ipairs(tree) do + line = assert(self:tree_to_nomsu(line, nil, true), "Could not convert line to nomsu") + nomsu:append(line) + if i < #tree then + nomsu:append("\n") + if match(tostring(line), "\n") then + nomsu:append("\n") + end + end + end + return nomsu + elseif "Text" == _exp_0 then + if inline then + local nomsu = NomsuCode(tree.source, '"') + for _index_0 = 1, #tree do + local bit = tree[_index_0] + if type(bit) == 'string' then + nomsu:append((gsub(gsub(gsub(bit, "\\", "\\\\"), "\n", "\\n"), '"', '\\"'))) + else + local interp_nomsu = self:tree_to_nomsu(bit, true) + if interp_nomsu then + if bit.type ~= "Var" and bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" then + interp_nomsu:parenthesize() + end + nomsu:append("\\", interp_nomsu) + else + return nil + end + end + end + nomsu:append('"') + return nomsu + else + local inline_version = self:tree_to_nomsu(tree, true) + if inline_version and #inline_version <= MAX_LINE then + return inline_version + end + local nomsu = NomsuCode(tree.source, '".."\n ') + for i, bit in ipairs(tree) do + if type(bit) == 'string' then + local bit_lines = get_lines:match(bit) + for j, line in ipairs(bit_lines) do + if j > 1 then + nomsu:append("\n ") + end + if #line > 1.25 * MAX_LINE then + local remainder = line + while #remainder > 0 do + local split = find(remainder, " ", MAX_LINE, true) + if split then + local chunk + chunk, remainder = sub(remainder, 1, split), sub(remainder, split + 1, -1) + nomsu:append(chunk) + elseif #remainder > 1.75 * MAX_LINE then + split = math.floor(1.5 * MAX_LINE) + local chunk + chunk, remainder = sub(remainder, 1, split), sub(remainder, split + 1, -1) + nomsu:append(chunk) + else + nomsu:append(remainder) + break + end + if #remainder > 0 then + nomsu:append("\\\n ..") + end + end + else + nomsu:append(line) + end + end + else + local interp_nomsu = self:tree_to_nomsu(bit, true) + if interp_nomsu then + if bit.type ~= "Var" and bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" then + interp_nomsu:parenthesize() + end + nomsu:append("\\", interp_nomsu) + else + interp_nomsu = assert(self:tree_to_nomsu(bit)) + if not (interp_nomsu) then + return nil + end + nomsu:append("\\\n ", interp_nomsu) + if i < #tree then + nomsu:append("\n ..") + end + end + end + end + return nomsu + end + elseif "List" == _exp_0 then + if inline then + local nomsu = NomsuCode(tree.source, "[") + for i, item in ipairs(tree) do + local item_nomsu = self:tree_to_nomsu(item, true) + if not (item_nomsu) then + return nil + end + if i > 1 then + nomsu:append(", ") + end + nomsu:append(item_nomsu) + end + nomsu:append("]") + return nomsu + else + local inline_version = self:tree_to_nomsu(tree, true) + if inline_version and #inline_version <= MAX_LINE then + return inline_version + end + local nomsu = NomsuCode(tree.source, "[..]") + local line = NomsuCode(tree.source, "\n ") + for _index_0 = 1, #tree do + local item = tree[_index_0] + local item_nomsu = self:tree_to_nomsu(item, true) + if item_nomsu and #line + #", " + #item_nomsu <= MAX_LINE then + if #line.bits > 1 then + line:append(", ") + end + line:append(item_nomsu) + else + if not (item_nomsu) then + item_nomsu = self:tree_to_nomsu(item) + if not (item_nomsu) then + return nil + end + end + if #line.bits > 1 then + nomsu:append(line) + line = NomsuCode(line.source, "\n ") + end + line:append(item_nomsu) + end + end + if #line.bits > 1 then + nomsu:append(line) + end + return nomsu + end + elseif "Dict" == _exp_0 then + if inline then + local nomsu = NomsuCode(tree.source, "{") + for i, entry in ipairs(tree) do + local entry_nomsu = self:tree_to_nomsu(entry, true) + if not (entry_nomsu) then + return nil + end + if i > 1 then + nomsu:append(", ") + end + nomsu:append(entry_nomsu) + end + nomsu:append("}") + return nomsu + else + local inline_version = self:tree_to_nomsu(tree, true) + if inline_version then + return inline_version + end + local nomsu = NomsuCode(tree.source, "{..}") + local line = NomsuCode(tree.source, "\n ") + for _index_0 = 1, #tree do + local entry = tree[_index_0] + local entry_nomsu = self:tree_to_nomsu(entry) + if not (entry_nomsu) then + return nil + end + if #line + #tostring(entry_nomsu) <= MAX_LINE then + if #line.bits > 1 then + line:append(", ") + end + line:append(entry_nomsu) + else + if #line.bits > 1 then + nomsu:append(line) + line = NomsuCode(line.source, "\n ") + end + line:append(entry_nomsu) + end + end + if #line.bits > 1 then + nomsu:append(line) + end + return nomsu + end + elseif "DictEntry" == _exp_0 then + local key, value = tree[1], tree[2] + local key_nomsu = self:tree_to_nomsu(key, true) + if not (key_nomsu) then + return nil + end + if key.type == "Action" or key.type == "Block" then + key_nomsu:parenthesize() + end + local value_nomsu + if value then + value_nomsu = self:tree_to_nomsu(value, true) + else + value_nomsu = NomsuCode(tree.source, "") + end + if inline and not value_nomsu then + return nil + end + if not value_nomsu then + if inline then + return nil + end + value_nomsu = self:tree_to_nomsu(value) + if not (value_nomsu) then + return nil + end + end + return NomsuCode(tree.source, key_nomsu, ":", value_nomsu) + elseif "IndexChain" == _exp_0 then + local nomsu = NomsuCode(tree.source) + for i, bit in ipairs(tree) do + if i > 1 then + nomsu:append(".") + end + local bit_nomsu + if bit.type == "Text" and #bit == 1 and type(bit[1]) == 'string' then + if bit[1]:match("[_a-zA-Z][_a-zA-Z0-9]*") then + bit_nomsu = bit[1] + end + end + if not (bit_nomsu) then + bit_nomsu = self:tree_to_nomsu(bit, true) + end + if not (bit_nomsu) then + return nil + end + local _exp_1 = bit.type + if "Action" == _exp_1 or "Block" == _exp_1 or "IndexChain" == _exp_1 then + bit_nomsu:parenthesize() + elseif "Number" == _exp_1 then + if i < #tree then + bit_nomsu:parenthesize() + end + end + nomsu:append(bit_nomsu) + end + return nomsu + elseif "Number" == _exp_0 then + return NomsuCode(tree.source, tostring(tree[1])) + elseif "Var" == _exp_0 then + return NomsuCode(tree.source, "%", tree[1]) + else + return error("Unknown type: " .. tostring(tree.type)) + end + end +end +return NomsuCompiler diff --git a/nomsu_compiler.moon b/nomsu_compiler.moon new file mode 100644 index 0000000..8226db4 --- /dev/null +++ b/nomsu_compiler.moon @@ -0,0 +1,797 @@ +-- This file contains the source code of the Nomsu compiler. +-- Nomsu is a programming language that cross-compiles to Lua. It was designed to be good +-- at natural-language-like code that is highly self-modifying and flexible. +-- The only dependency is LPEG, which can be installed using "luarocks install lpeg" +-- File usage: +-- Either, in a lua/moonscript file: +-- Nomsu = require "nomsu" +-- nomsu = Nomsu() +-- nomsu:run(your_nomsu_code) +-- Or from the command line: +-- lua nomsu.lua your_file.nom +lpeg = require 'lpeg' +re = require 're' +lpeg.setmaxstack 10000 +utils = require 'utils' +{:repr, :stringify, :equivalent} = utils +export colors, colored +colors = require 'consolecolors' +colored = setmetatable({}, {__index:(_,color)-> ((msg)-> colors[color]..tostring(msg or '')..colors.reset)}) +{:insert, :remove, :concat} = table +unpack or= table.unpack +{:match, :sub, :rep, :gsub, :format, :byte, :match, :find} = string +{:NomsuCode, :LuaCode, :Source} = require "code_obj" +AST = require "nomsu_tree" +parse = require("parser") +-- Mapping from source string (e.g. "@core/metaprogramming.nom[1:100]") to a mapping +-- from lua line number to nomsu line number +export SOURCE_MAP +SOURCE_MAP = {} + +string.as_lua_id = (str)-> + argnum = 0 + -- Cut up escape-sequence-like chunks + str = gsub str, "x([0-9A-F][0-9A-F])", "x\0%1" + -- Alphanumeric unchanged, spaces to underscores, and everything else to hex escape sequences + str = gsub str, "%W", (c)-> + if c == ' ' then '_' + elseif c == '%' then + argnum += 1 + tostring(argnum) + else format("x%02X", byte(c)) + return '_'..str + +table.map = (fn)=> [fn(v) for _,v in ipairs(@)] +table.fork = (t, values)-> setmetatable(values or {}, {__index:t}) + +-- TODO: +-- consider non-linear codegen, rather than doing thunks for things like comprehensions +-- Add a ((%x foo %y) where {x:"asdf", y:"fdsa"}) compile-time action for substitution +-- Re-implement nomsu-to-lua comment translation? + +export FILE_CACHE +-- FILE_CACHE is a map from filename (string) -> string of file contents +FILE_CACHE = setmetatable {}, { + __index: (filename)=> + file = io.open(filename) + return nil unless file + contents = file\read("*a") + file\close! + self[filename] = contents + return contents +} + +export all_files +iterate_single = (item, prev) -> if item == prev then nil else item +all_files = (path)-> + -- Sanitize path + if match(path, "%.nom$") or match(path, "%.lua$") or match(path, "^/dev/fd/[012]$") + return iterate_single, path + -- TODO: improve sanitization + path = gsub(path,"\\","\\\\") + path = gsub(path,"`","") + path = gsub(path,'"','\\"') + path = gsub(path,"$","") + return coroutine.wrap -> + f = io.popen('find -L "'..path..'" -not -path "*/\\.*" -type f -name "*.nom"') + for line in f\lines! + coroutine.yield(line) + success = f\close! + unless success + error("Invalid file path: "..tostring(path)) + +line_counter = re.compile([[ + lines <- {| line (%nl line)* |} + line <- {} (!%nl .)* +]], nl:lpeg.P("\r")^-1 * lpeg.P("\n")) +get_lines = re.compile([[ + lines <- {| line (%nl line)* |} + line <- {[^%nl]*} +]], nl:lpeg.P("\r")^-1 * lpeg.P("\n")) +-- Mapping from line number -> character offset +export LINE_STARTS +-- LINE_STARTS is a mapping from strings to a table that maps line number to character positions +LINE_STARTS = setmetatable {}, { + __mode:"k" + __index: (k)=> + -- Implicitly convert Lua and Nomsu objects to strings + if type(k) != 'string' + k = tostring(k) + if v = rawget(self, k) + return v + line_starts = line_counter\match(k) + self[k] = line_starts + return line_starts +} +export pos_to_line +pos_to_line = (str, pos)-> + line_starts = LINE_STARTS[str] + -- Binary search for line number of position + lo, hi = 1, #line_starts + while lo <= hi + mid = math.floor((lo+hi)/2) + if line_starts[mid] > pos + hi = mid-1 + else lo = mid+1 + return hi + +-- Use + operator for string coercive concatenation (note: "asdf" + 3 == "asdf3") +-- Use [] for accessing string characters, or s[{3,4}] for s:sub(3,4) +-- Note: This globally affects all strings in this instance of Lua! +do + STRING_METATABLE = getmetatable("") + STRING_METATABLE.__add = (other)=> @ .. stringify(other) + STRING_METATABLE.__index = (i)=> + ret = string[i] + if ret != nil then return ret + if type(i) == 'number' then return sub(@, i, i) + elseif type(i) == 'table' then return sub(@, i[1], i[2]) + +-- List and Dict classes to provide basic equality/tostring functionality for the tables +-- used in Nomsu. This way, they retain a notion of whether they were originally lists or dicts. +_list_mt = + __eq:equivalent + -- Could consider adding a __newindex to enforce list-ness, but would hurt performance + __tostring: => + "["..concat([repr(b) for b in *@], ", ").."]" +list = (t)-> setmetatable(t, _list_mt) + +_dict_mt = + __eq:equivalent + __tostring: => + "{"..concat(["#{repr(k)}: #{repr(v)}" for k,v in pairs @], ", ").."}" +dict = (t)-> setmetatable(t, _dict_mt) + +MAX_LINE = 80 -- For beautification purposes, try not to make lines much longer than this value +NomsuCompiler = setmetatable({}, {__index: (k)=> if _self = rawget(@, "self") then _self[k] else nil}) +with NomsuCompiler + ._ENV = NomsuCompiler + .nomsu = NomsuCompiler + .parse = (...)=> parse(...) + + -- Discretionary/convenience stuff + to_add = { + repr:repr, stringify:stringify, utils:utils, lpeg:lpeg, re:re, + -- Lua stuff: + :next, :unpack, :setmetatable, :coroutine, :rawequal, :getmetatable, :pcall, + :error, :package, :os, :require, :tonumber, :tostring, :string, :xpcall, :module, + :print, :loadfile, :rawset, :_VERSION, :collectgarbage, :rawget, :rawlen, + :table, :assert, :dofile, :loadstring, :type, :select, :debug, :math, :io, :load, + :pairs, :ipairs, + -- Nomsu types: + :list, :dict, + } + for k,v in pairs(to_add) do NomsuCompiler[k] = v + for k,v in pairs(AST) do NomsuCompiler[k] = v + .LuaCode = LuaCode + .NomsuCode = NomsuCode + .Source = Source + .ALIASES = setmetatable({}, {__mode:"k"}) + .LOADED = {} + .AST = AST + + .compile_error = (tok, err_format_string, ...)=> + file = FILE_CACHE[tok.source.filename] + line_no = pos_to_line(file, tok.source.start) + line_start = LINE_STARTS[file][line_no] + src = colored.dim(file\sub(line_start, tok.source.start-1)) + src ..= colored.underscore colored.bright colored.red(file\sub(tok.source.start, tok.source.stop-1)) + end_of_line = (LINE_STARTS[file][pos_to_line(file, tok.source.stop) + 1] or 0) - 1 + src ..= colored.dim(file\sub(tok.source.stop, end_of_line-1)) + src = ' '..src\gsub('\n', '\n ') + err_msg = err_format_string\format(src, ...) + error("#{tok.source.filename}:#{line_no}: "..err_msg, 0) + + -- This is a bit of a hack, but this code handles arbitrarily complex + -- math expressions like 2*x + 3^2 without having to define a single + -- action for every possibility. + math_expression = re.compile [[ ([+-] " ")* "%" (" " [*/^+-] (" " [+-])* " %")+ !. ]] + add_lua_bits = (lua, code)=> + for bit in *code + if type(bit) == "string" + lua\append bit + else + bit_lua = @compile(bit) + unless bit_lua.is_value + @compile_error bit, + "Cannot use:\n%s\nas a string interpolation value, since it's not an expression." + lua\append bit_lua + return lua + + add_lua_string_bits = (lua, code)=> + line_len = 0 + if code.type != "Text" + lua\append ", ", @compile(code) + return + for bit in *code + bit_lua = if type(bit) == "string" + repr(bit) + else + bit_lua = @compile(bit) + unless bit_lua.is_value + @compile_error bit, + "Cannot use:\n%s\nas a string interpolation value, since it's not an expression." + bit_lua + line_len += #tostring(bit_lua) + if line_len > MAX_LINE + lua\append ",\n " + line_len = 4 + else + lua\append ", " + lua\append bit_lua + + .COMPILE_ACTIONS = setmetatable { + ["# compile math expr #"]: (tree, ...)=> + lua = LuaCode.Value(tree.source) + for i,tok in ipairs tree + if type(tok) == 'string' + lua\append tok + else + tok_lua = @compile(tok) + unless tok_lua.is_value + @compile_error tok, "Non-expression value inside math expression:\n%s" + if tok.type == "Action" + tok_lua\parenthesize! + lua\append tok_lua + if i < #tree + lua\append " " + return lua + + ["Lua %"]: (tree, _code)=> + lua = LuaCode.Value(_code.source, "LuaCode(", repr(tostring _code.source)) + add_lua_string_bits(@, lua, _code) + lua\append ")" + return lua + + ["Lua value %"]: (tree, _code)=> + lua = LuaCode.Value(_code.source, "LuaCode.Value(", repr(tostring _code.source)) + add_lua_string_bits(@, lua, _code) + lua\append ")" + return lua + + ["lua > %"]: (tree, _code)=> + if _code.type != "Text" + return LuaCode tree.source, "nomsu:run_lua(", @compile(_code), ");" + return add_lua_bits(@, LuaCode(tree.source), _code) + + ["= lua %"]: (tree, _code)=> + if _code.type != "Text" + return LuaCode.Value tree.source, "nomsu:run_lua(", @compile(_code), ":as_statements('return '))" + return add_lua_bits(@, LuaCode.Value(tree.source), _code) + + ["use %"]: (tree, _path)=> + unless _path.type == 'Text' and #_path == 1 and type(_path[1]) == 'string' + return LuaCode(tree.source, "nomsu:run_file(#{@compile(_path)});") + path = _path[1] + @run_file(path) + return LuaCode(tree.source, "nomsu:run_file(#{repr path});") + }, { + __index: (stub)=> + if math_expression\match(stub) + return @["# compile math expr #"] + } + + .run = (to_run, source=nil)=> + tree = if AST.is_syntax_tree(to_run) then tree else @parse(to_run, source or to_run.source) + if tree == nil -- Happens if pattern matches, but there are no captures, e.g. an empty string + return nil + if tree.type == "FileChunks" + -- Each chunk's compilation is affected by the code in the previous chunks + -- (typically), so each chunk needs to compile and run before the next one + -- compiles. + ret = nil + all_lua = {} + for chunk in *tree + lua = @compile(chunk)\as_statements! + lua\declare_locals! + lua\prepend "-- File: #{chunk.source or ""}\n" + insert all_lua, tostring(lua) + ret = @run_lua(lua) + if @on_compile + self.on_compile(concat(all_lua, "\n"), (source or to_run.source).filename) + return ret + else + lua = @compile(tree, compile_actions)\as_statements! + lua\declare_locals! + lua\prepend "-- File: #{source or to_run.source or ""}\n" + if @on_compile + self.on_compile(lua, (source or to_run.source).filename) + return @run_lua(lua) + + _running_files = {} -- For detecting circular imports + .run_file = (filename)=> + if @LOADED[filename] + return @LOADED[filename] + ret = nil + for filename in all_files(filename) + if ret = @LOADED[filename] + continue + + -- Check for circular import + for i,running in ipairs _running_files + if running == filename + loop = [_running_files[j] for j=i,#_running_files] + insert loop, filename + error("Circular import, this loops forever: #{concat loop, " -> "}...") + + insert _running_files, filename + if match(filename, "%.lua$") + file = assert(FILE_CACHE[filename], "Could not find file: #{filename}") + ret = @run_lua file, Source(filename, 1, #file) + elseif match(filename, "%.nom$") or match(filename, "^/dev/fd/[012]$") + ran_lua = if not @skip_precompiled -- Look for precompiled version + lua_filename = gsub(filename, "%.nom$", ".lua") + if file = FILE_CACHE[lua_filename] + ret = @run_lua file, Source(lua_filename, 1, #file) + true + unless ran_lua + file = file or FILE_CACHE[filename] + if not file + error("File does not exist: #{filename}", 0) + ret = @run file, Source(filename,1,#file) + else + error("Invalid filetype for #{filename}", 0) + @LOADED[filename] = ret or true + remove _running_files + + @LOADED[filename] = ret or true + return ret + + .run_lua = (lua, source=nil)=> + lua_string = tostring(lua) + run_lua_fn, err = load(lua_string, nil and tostring(source or lua.source), "t", self) + if not run_lua_fn + line_numbered_lua = concat( + [format("%3d|%s",i,line) for i, line in ipairs get_lines\match(lua_string)], + "\n") + error("Failed to compile generated code:\n#{colored.bright colored.blue colored.onblack line_numbered_lua}\n\n#{err}", 0) + source_key = tostring(source or lua.source) + unless SOURCE_MAP[source_key] + map = {} + offset = 1 + source or= lua.source + nomsu_str = tostring(FILE_CACHE[source.filename]\sub(source.start, source.stop)) + lua_line = 1 + nomsu_line = pos_to_line(nomsu_str, source.start) + fn = (s)-> + if type(s) == 'string' + for nl in s\gmatch("\n") + map[lua_line] or= nomsu_line + lua_line += 1 + else + old_line = nomsu_line + if s.source + nomsu_line = pos_to_line(nomsu_str, s.source.start) + for b in *s.bits do fn(b) + fn(lua) + map[lua_line] or= nomsu_line + map[0] = 0 + -- Mapping from lua line number to nomsu line numbers + SOURCE_MAP[source_key] = map + + return run_lua_fn! + + .compile = (tree)=> + switch tree.type + when "Action" + stub = tree.stub + if compile_action = @COMPILE_ACTIONS[stub] + args = [arg for arg in *tree when type(arg) != "string"] + -- Force Lua to avoid tail call optimization for debugging purposes + -- TODO: use tail call? + ret = compile_action(@, tree, unpack(args)) + if not ret + @compile_error tree, + "Compile-time action:\n%s\nfailed to produce any Lua" + return ret + + lua = LuaCode.Value(tree.source, "A",string.as_lua_id(stub),"(") + args = {} + for i, tok in ipairs tree + if type(tok) == "string" then continue + arg_lua = @compile(tok) + unless arg_lua.is_value + @compile_error tok, + "Cannot use:\n%s\nas an argument to %s, since it's not an expression, it produces: %s", + stub, repr arg_lua + insert args, arg_lua + lua\concat_append args, ", " + lua\append ")" + return lua + + when "EscapedNomsu" + make_tree = (t)-> + unless AST.is_syntax_tree(t) + return repr(t) + bits = [make_tree(bit) for bit in *t] + return t.type.."("..repr(tostring t.source)..", "..table.concat(bits, ", ")..")" + LuaCode.Value tree.source, make_tree(tree[1]) + + when "Block" + lua = LuaCode(tree.source) + lua\concat_append([@compile(line)\as_statements! for line in *tree], "\n") + return lua + + when "Text" + lua = LuaCode.Value(tree.source) + string_buffer = "" + for i, bit in ipairs tree + if type(bit) == "string" + string_buffer ..= bit + continue + if string_buffer ~= "" + if #lua.bits > 0 then lua\append ".." + lua\append repr(string_buffer) + string_buffer = "" + bit_lua = @compile(bit) + unless bit_lua.is_value + src = ' '..gsub(tostring(@tree_to_nomsu(bit)), '\n','\n ') + line = "#{bit.source.filename}:#{pos_to_line(FILE_CACHE[bit.source.filename], bit.source.start)}" + @compile_error bit, + "Cannot use:\n%s\nas a string interpolation value, since it's not an expression." + if #lua.bits > 0 then lua\append ".." + if bit.type != "Text" + bit_lua = LuaCode.Value(bit.source, "stringify(",bit_lua,")") + lua\append bit_lua + + if string_buffer ~= "" or #lua.bits == 0 + if #lua.bits > 0 then lua\append ".." + lua\append repr(string_buffer) + + if #lua.bits > 1 + lua\parenthesize! + return lua + + when "List" + lua = LuaCode.Value tree.source, "list{" + items = {} + for i, item in ipairs tree + item_lua = @compile(item) + unless item_lua.is_value + @compile_error item, + "Cannot use:\n%s\nas a list item, since it's not an expression." + items[i] = item_lua + lua\concat_append(items, ", ", ",\n ") + lua\append "}" + return lua + + when "Dict" + lua = LuaCode.Value tree.source, "dict{" + lua\concat_append([@compile(e) for e in *tree], ", ", ",\n ") + lua\append "}" + return lua + + when "DictEntry" + key, value = tree[1], tree[2] + key_lua = @compile(key) + unless key_lua.is_value + @compile_error tree[1], + "Cannot use:\n%s\nas a dict key, since it's not an expression." + value_lua = value and @compile(value) or LuaCode.Value(key.source, "true") + unless value_lua.is_value + @compile_error tree[2], + "Cannot use:\n%s\nas a dict value, since it's not an expression." + -- TODO: support arbitrary words here, like operators and unicode + key_str = match(tostring(key_lua), [=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=]) + return if key_str + LuaCode tree.source, key_str,"=",value_lua + elseif sub(tostring(key_lua),1,1) == "[" + -- NOTE: this *must* use a space after the [ to avoid freaking out + -- Lua's parser if the inner expression is a long string. Lua + -- parses x[[[y]]] as x("[y]"), not as x["y"] + LuaCode tree.source, "[ ",key_lua,"]=",value_lua + else + LuaCode tree.source, "[",key_lua,"]=",value_lua + + when "IndexChain" + lua = @compile(tree[1]) + unless lua.is_value + @compile_error tree[1], + "Cannot index:\n%s\nsince it's not an expression." + first_char = sub(tostring(lua),1,1) + if first_char == "{" or first_char == '"' or first_char == "[" + lua\parenthesize! + + for i=2,#tree + key = tree[i] + key_lua = @compile(key) + unless key_lua.is_value + @compile_error key, + "Cannot use:\n%s\nas an index, since it's not an expression." + key_lua_str = tostring(key_lua) + if lua_id = match(key_lua_str, "^['\"]([a-zA-Z_][a-zA-Z0-9_]*)['\"]$") + lua\append ".#{lua_id}" + elseif sub(key_lua_str,1,1) == '[' + -- NOTE: this *must* use a space after the [ to avoid freaking out + -- Lua's parser if the inner expression is a long string. Lua + -- parses x[[[y]]] as x("[y]"), not as x["y"] + lua\append "[ ",key_lua," ]" + else + lua\append "[",key_lua,"]" + return lua + + when "Number" + LuaCode.Value(tree.source, tostring(tree[1])) + + when "Var" + LuaCode.Value(tree.source, string.as_lua_id(tree[1])) + + when "FileChunks" + error("Cannot convert FileChunks to a single block of lua, since each chunk's ".. + "compilation depends on the earlier chunks") + + else + error("Unknown type: #{tree.type}") + + .tree_to_nomsu = (tree, inline=false, can_use_colon=false)=> + switch tree.type + when "FileChunks" + return nil if inline + nomsu = NomsuCode(tree.source) + nomsu\concat_append [@tree_to_nomsu(c) for c in *tree], "\n#{("~")\rep(80)}\n" + return nomsu + + when "Action" + if inline + nomsu = NomsuCode(tree.source) + for i,bit in ipairs tree + if type(bit) == "string" + if i > 1 + nomsu\append " " + nomsu\append bit + else + arg_nomsu = @tree_to_nomsu(bit,true) + return nil unless arg_nomsu + unless i == 1 + nomsu\append " " + if bit.type == "Action" or bit.type == "Block" + arg_nomsu\parenthesize! + nomsu\append arg_nomsu + return nomsu + else + nomsu = NomsuCode(tree.source) + next_space = "" + line_len, last_colon = 0, nil + for i,bit in ipairs tree + if type(bit) == "string" + line_len += #next_space + #bit + nomsu\append next_space, bit + next_space = " " + else + arg_nomsu = if last_colon == i-1 and bit.type == "Action" then nil + elseif bit.type == "Block" then nil + else @tree_to_nomsu(bit,true) + + if arg_nomsu and line_len + #tostring(arg_nomsu) < MAX_LINE + if bit.type == "Action" + if can_use_colon and i > 1 + nomsu\append match(next_space,"[^ ]*"), ": ", arg_nomsu + next_space = "\n.." + line_len = 2 + last_colon = i + else + nomsu\append next_space, "(", arg_nomsu, ")" + line_len += #next_space + 2 + #tostring(arg_nomsu) + next_space = " " + else + nomsu\append next_space, arg_nomsu + line_len += #next_space + #tostring(arg_nomsu) + next_space = " " + else + arg_nomsu = @tree_to_nomsu(bit, nil, true) + return nil unless nomsu + -- These types carry their own indentation + if bit.type != "List" and bit.type != "Dict" and bit.type != "Text" + if i == 1 + arg_nomsu = NomsuCode(bit.source, "(..)\n ", arg_nomsu) + else + arg_nomsu = NomsuCode(bit.source, "\n ", arg_nomsu) + + if last_colon == i-1 and (bit.type == "Action" or bit.type == "Block") + next_space = "" + nomsu\append next_space, arg_nomsu + next_space = "\n.." + line_len = 2 + + if next_space == " " and #(match(tostring(nomsu),"[^\n]*$")) > MAX_LINE + next_space = "\n.." + return nomsu + + when "EscapedNomsu" + nomsu = @tree_to_nomsu(tree[1], true) + if nomsu == nil and not inline + nomsu = @tree_to_nomsu(tree[1]) + return nomsu and NomsuCode tree.source, "\\:\n ", nomsu + return nomsu and NomsuCode tree.source, "\\(", nomsu, ")" + + when "Block" + if inline + nomsu = NomsuCode(tree.source) + for i,line in ipairs tree + if i > 1 + nomsu\append "; " + line_nomsu = @tree_to_nomsu(line,true) + return nil unless line_nomsu + nomsu\append line_nomsu + return nomsu + nomsu = NomsuCode(tree.source) + for i, line in ipairs tree + line = assert(@tree_to_nomsu(line, nil, true), "Could not convert line to nomsu") + nomsu\append line + if i < #tree + nomsu\append "\n" + if match(tostring(line), "\n") + nomsu\append "\n" + return nomsu + + when "Text" + if inline + nomsu = NomsuCode(tree.source, '"') + for bit in *tree + if type(bit) == 'string' + -- TODO: unescape better? + nomsu\append (gsub(gsub(gsub(bit,"\\","\\\\"),"\n","\\n"),'"','\\"')) + else + interp_nomsu = @tree_to_nomsu(bit, true) + if interp_nomsu + if bit.type != "Var" and bit.type != "List" and bit.type != "Dict" and bit.type != "Text" + interp_nomsu\parenthesize! + nomsu\append "\\", interp_nomsu + else return nil + nomsu\append '"' + return nomsu + else + inline_version = @tree_to_nomsu(tree, true) + if inline_version and #inline_version <= MAX_LINE + return inline_version + nomsu = NomsuCode(tree.source, '".."\n ') + for i, bit in ipairs tree + if type(bit) == 'string' + bit_lines = get_lines\match(bit) + for j, line in ipairs bit_lines + if j > 1 then nomsu\append "\n " + if #line > 1.25*MAX_LINE + remainder = line + while #remainder > 0 + split = find(remainder, " ", MAX_LINE, true) + if split + chunk, remainder = sub(remainder, 1, split), sub(remainder, split+1, -1) + nomsu\append chunk + elseif #remainder > 1.75*MAX_LINE + split = math.floor(1.5*MAX_LINE) + chunk, remainder = sub(remainder, 1, split), sub(remainder, split+1, -1) + nomsu\append chunk + else + nomsu\append remainder + break + if #remainder > 0 then nomsu\append "\\\n .." + else + nomsu\append line + else + interp_nomsu = @tree_to_nomsu(bit, true) + if interp_nomsu + if bit.type != "Var" and bit.type != "List" and bit.type != "Dict" and bit.type != "Text" + interp_nomsu\parenthesize! + nomsu\append "\\", interp_nomsu + else + interp_nomsu = assert(@tree_to_nomsu(bit)) + return nil unless interp_nomsu + nomsu\append "\\\n ", interp_nomsu + if i < #tree + nomsu\append "\n .." + return nomsu + + when "List" + if inline + nomsu = NomsuCode(tree.source, "[") + for i, item in ipairs tree + item_nomsu = @tree_to_nomsu(item, true) + return nil unless item_nomsu + if i > 1 + nomsu\append ", " + nomsu\append item_nomsu + nomsu\append "]" + return nomsu + else + inline_version = @tree_to_nomsu(tree, true) + if inline_version and #inline_version <= MAX_LINE + return inline_version + nomsu = NomsuCode(tree.source, "[..]") + line = NomsuCode(tree.source, "\n ") + for item in *tree + item_nomsu = @tree_to_nomsu(item, true) + if item_nomsu and #line + #", " + #item_nomsu <= MAX_LINE + if #line.bits > 1 + line\append ", " + line\append item_nomsu + else + unless item_nomsu + item_nomsu = @tree_to_nomsu(item) + return nil unless item_nomsu + if #line.bits > 1 + nomsu\append line + line = NomsuCode(line.source, "\n ") + line\append item_nomsu + if #line.bits > 1 + nomsu\append line + return nomsu + + when "Dict" + if inline + nomsu = NomsuCode(tree.source, "{") + for i, entry in ipairs tree + entry_nomsu = @tree_to_nomsu(entry, true) + return nil unless entry_nomsu + if i > 1 + nomsu\append ", " + nomsu\append entry_nomsu + nomsu\append "}" + return nomsu + else + inline_version = @tree_to_nomsu(tree, true) + if inline_version then return inline_version + nomsu = NomsuCode(tree.source, "{..}") + line = NomsuCode(tree.source, "\n ") + for entry in *tree + entry_nomsu = @tree_to_nomsu(entry) + return nil unless entry_nomsu + if #line + #tostring(entry_nomsu) <= MAX_LINE + if #line.bits > 1 + line\append ", " + line\append entry_nomsu + else + if #line.bits > 1 + nomsu\append line + line = NomsuCode(line.source, "\n ") + line\append entry_nomsu + if #line.bits > 1 + nomsu\append line + return nomsu + + when "DictEntry" + key, value = tree[1], tree[2] + key_nomsu = @tree_to_nomsu(key, true) + return nil unless key_nomsu + if key.type == "Action" or key.type == "Block" + key_nomsu\parenthesize! + value_nomsu = if value + @tree_to_nomsu(value, true) + else NomsuCode(tree.source, "") + if inline and not value_nomsu then return nil + if not value_nomsu + return nil if inline + value_nomsu = @tree_to_nomsu(value) + return nil unless value_nomsu + return NomsuCode tree.source, key_nomsu, ":", value_nomsu + + when "IndexChain" + nomsu = NomsuCode(tree.source) + for i, bit in ipairs tree + if i > 1 + nomsu\append "." + local bit_nomsu + if bit.type == "Text" and #bit == 1 and type(bit[1]) == 'string' + -- TODO: support arbitrary words here, including operators and unicode + if bit[1]\match("[_a-zA-Z][_a-zA-Z0-9]*") + bit_nomsu = bit[1] + unless bit_nomsu then bit_nomsu = @tree_to_nomsu(bit, true) + return nil unless bit_nomsu + switch bit.type + when "Action", "Block", "IndexChain" + bit_nomsu\parenthesize! + when "Number" + if i < #tree + bit_nomsu\parenthesize! + nomsu\append bit_nomsu + return nomsu + + when "Number" + return NomsuCode(tree.source, tostring(tree[1])) + + when "Var" + return NomsuCode(tree.source, "%", tree[1]) + + else + error("Unknown type: #{tree.type}") + +return NomsuCompiler