2018-11-08 15:23:22 -08:00
|
|
|
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 Importer, import_to_1_from, _1_forked
|
|
|
|
do
|
|
|
|
local _obj_0 = require('importer')
|
|
|
|
Importer, import_to_1_from, _1_forked = _obj_0.Importer, _obj_0.import_to_1_from, _obj_0._1_forked
|
|
|
|
end
|
|
|
|
local List, Dict, Text
|
|
|
|
do
|
|
|
|
local _obj_0 = require('containers')
|
|
|
|
List, Dict, Text = _obj_0.List, _obj_0.Dict, _obj_0.Text
|
|
|
|
end
|
|
|
|
local SyntaxTree = require("syntax_tree")
|
|
|
|
local Files = require("files")
|
|
|
|
local make_parser = require("parser")
|
|
|
|
local pretty_error = require("pretty_errors")
|
|
|
|
local make_tree
|
|
|
|
make_tree = function(tree, userdata)
|
|
|
|
tree.source = Source(userdata.filename, tree.start, tree.stop)
|
|
|
|
tree.start, tree.stop = nil, nil
|
|
|
|
tree = SyntaxTree(tree)
|
|
|
|
return tree
|
|
|
|
end
|
|
|
|
local Parsers = { }
|
|
|
|
local max_parser_version = 0
|
|
|
|
for version = 1, 999 do
|
|
|
|
local peg_file
|
|
|
|
if package.nomsupath then
|
|
|
|
for path in package.nomsupath:gmatch("[^;]+") do
|
|
|
|
peg_file = io.open(path .. "/nomsu." .. tostring(version) .. ".peg")
|
|
|
|
if peg_file then
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
|
|
|
else
|
|
|
|
peg_file = io.open("nomsu." .. tostring(version) .. ".peg")
|
|
|
|
end
|
|
|
|
if not (peg_file) then
|
|
|
|
break
|
|
|
|
end
|
|
|
|
max_parser_version = version
|
|
|
|
local peg_contents = peg_file:read('*a')
|
|
|
|
peg_file:close()
|
|
|
|
Parsers[version] = make_parser(peg_contents, make_tree)
|
|
|
|
end
|
|
|
|
local tree_to_nomsu, tree_to_inline_nomsu
|
|
|
|
do
|
|
|
|
local _obj_0 = require("nomsu_decompiler")
|
|
|
|
tree_to_nomsu, tree_to_inline_nomsu = _obj_0.tree_to_nomsu, _obj_0.tree_to_inline_nomsu
|
|
|
|
end
|
2018-11-09 14:48:23 -08:00
|
|
|
local compile = require('nomsu_compiler')
|
2018-11-09 16:40:36 -08:00
|
|
|
local _currently_running_files = List({ })
|
2018-11-08 15:23:22 -08:00
|
|
|
local nomsu_environment = Importer({
|
|
|
|
NOMSU_COMPILER_VERSION = 12,
|
|
|
|
NOMSU_SYNTAX_VERSION = max_parser_version,
|
|
|
|
next = next,
|
|
|
|
unpack = unpack or table.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,
|
|
|
|
lua_type_of = type,
|
|
|
|
select = select,
|
|
|
|
math = math,
|
|
|
|
io = io,
|
|
|
|
load = load,
|
|
|
|
pairs = pairs,
|
|
|
|
ipairs = ipairs,
|
|
|
|
jit = jit,
|
|
|
|
_VERSION = _VERSION,
|
|
|
|
bit = (jit or _VERSION == "Lua 5.2") and require('bitops') or nil,
|
|
|
|
List = List,
|
|
|
|
Dict = Dict,
|
|
|
|
lpeg = lpeg,
|
|
|
|
re = re,
|
|
|
|
Files = Files,
|
|
|
|
SyntaxTree = SyntaxTree,
|
|
|
|
TESTS = Dict({ }),
|
|
|
|
globals = Dict({ }),
|
|
|
|
LuaCode = LuaCode,
|
|
|
|
NomsuCode = NomsuCode,
|
|
|
|
Source = Source,
|
|
|
|
SOURCE_MAP = Importer({ }),
|
|
|
|
_1_as_nomsu = tree_to_nomsu,
|
|
|
|
_1_as_inline_nomsu = tree_to_inline_nomsu,
|
2018-11-09 14:48:23 -08:00
|
|
|
compile = compile,
|
|
|
|
_1_as_lua = compile,
|
2018-11-08 15:23:22 -08:00
|
|
|
_1_forked = _1_forked,
|
|
|
|
import_to_1_from = import_to_1_from,
|
|
|
|
_1_parsed = function(nomsu_code)
|
|
|
|
if type(nomsu_code) == 'string' then
|
|
|
|
local filename = Files.spoof(nomsu_code)
|
2018-11-09 16:40:36 -08:00
|
|
|
nomsu_code = NomsuCode:from(Source(filename, 1, #nomsu_code), nomsu_code)
|
2018-11-08 15:23:22 -08:00
|
|
|
end
|
|
|
|
local source = nomsu_code.source
|
|
|
|
nomsu_code = tostring(nomsu_code)
|
|
|
|
local version = nomsu_code:match("^#![^\n]*nomsu[ ]+-V[ ]*([0-9.]+)")
|
|
|
|
local syntax_version = version and tonumber(version:match("^[0-9]+")) or max_parser_version
|
|
|
|
local parse = Parsers[syntax_version] or Parsers[max_parser_version]
|
|
|
|
local tree = parse(nomsu_code, source.filename)
|
2018-11-11 15:05:18 -08:00
|
|
|
if tree.shebang then
|
|
|
|
tree.version = tree.shebang:match("nomsu %-V[ ]*([%d.]*)")
|
|
|
|
end
|
2018-11-08 15:23:22 -08:00
|
|
|
local find_errors
|
|
|
|
find_errors = function(t)
|
|
|
|
if t.type == "Error" then
|
|
|
|
return coroutine.yield(t)
|
|
|
|
else
|
|
|
|
for k, v in pairs(t) do
|
|
|
|
local _continue_0 = false
|
|
|
|
repeat
|
|
|
|
if not (SyntaxTree:is_instance(v)) then
|
|
|
|
_continue_0 = true
|
|
|
|
break
|
|
|
|
end
|
|
|
|
find_errors(v)
|
|
|
|
_continue_0 = true
|
|
|
|
until true
|
|
|
|
if not _continue_0 then
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
local errs
|
|
|
|
do
|
|
|
|
local _accum_0 = { }
|
|
|
|
local _len_0 = 1
|
|
|
|
for err in coroutine.wrap(function()
|
|
|
|
return find_errors(tree)
|
|
|
|
end) do
|
|
|
|
_accum_0[_len_0] = err
|
|
|
|
_len_0 = _len_0 + 1
|
|
|
|
end
|
|
|
|
errs = _accum_0
|
|
|
|
end
|
|
|
|
local num_errs = #errs
|
|
|
|
if num_errs > 0 then
|
|
|
|
local err_strings
|
|
|
|
do
|
|
|
|
local _accum_0 = { }
|
|
|
|
local _len_0 = 1
|
|
|
|
for i, e in ipairs(errs) do
|
|
|
|
if i <= 3 then
|
|
|
|
_accum_0[_len_0] = pretty_error({
|
|
|
|
title = "Parse error",
|
|
|
|
error = e.error,
|
|
|
|
hint = e.hint,
|
2018-11-09 14:36:15 -08:00
|
|
|
source = e:get_source_file(),
|
2018-11-08 15:23:22 -08:00
|
|
|
start = e.source.start,
|
|
|
|
stop = e.source.stop,
|
|
|
|
filename = e.source.filename
|
|
|
|
})
|
|
|
|
_len_0 = _len_0 + 1
|
|
|
|
end
|
|
|
|
end
|
|
|
|
err_strings = _accum_0
|
|
|
|
end
|
|
|
|
if num_errs > #err_strings then
|
|
|
|
table.insert(err_strings, "\027[31;1m +" .. tostring(num_errs - #err_strings) .. " additional errors...\027[0m\n")
|
|
|
|
end
|
|
|
|
error(table.concat(err_strings, '\n\n'), 0)
|
|
|
|
end
|
|
|
|
return tree
|
|
|
|
end,
|
|
|
|
run_1_in = function(to_run, environment)
|
|
|
|
if type(to_run) == 'string' then
|
|
|
|
local filename = Files.spoof(to_run)
|
2018-11-09 16:40:36 -08:00
|
|
|
to_run = NomsuCode:from(Source(filename, 1, #to_run), to_run)
|
2018-11-08 15:23:22 -08:00
|
|
|
local ret = environment.run_1_in(to_run, environment)
|
|
|
|
return ret
|
|
|
|
elseif NomsuCode:is_instance(to_run) then
|
|
|
|
local tree = environment._1_parsed(to_run)
|
|
|
|
if tree == nil then
|
|
|
|
return nil
|
|
|
|
end
|
|
|
|
local ret = environment.run_1_in(tree, environment)
|
|
|
|
return ret
|
|
|
|
elseif SyntaxTree:is_instance(to_run) then
|
|
|
|
local filename = to_run.source.filename:gsub("\n.*", "...")
|
|
|
|
if to_run.type ~= "FileChunks" then
|
|
|
|
to_run = {
|
|
|
|
to_run
|
|
|
|
}
|
|
|
|
end
|
|
|
|
local ret = nil
|
|
|
|
for _index_0 = 1, #to_run do
|
|
|
|
local chunk = to_run[_index_0]
|
|
|
|
local lua = environment.compile(chunk)
|
|
|
|
lua:declare_locals()
|
|
|
|
lua:prepend("-- File: " .. tostring(filename) .. "\n")
|
|
|
|
ret = environment.run_1_in(lua, environment)
|
|
|
|
end
|
|
|
|
return ret
|
|
|
|
elseif LuaCode:is_instance(to_run) then
|
|
|
|
local source = to_run.source
|
|
|
|
local lua_string = to_run:text()
|
|
|
|
local run_lua_fn, err = load(lua_string, tostring(source), "t", environment)
|
|
|
|
if not run_lua_fn then
|
|
|
|
local lines
|
|
|
|
do
|
|
|
|
local _accum_0 = { }
|
|
|
|
local _len_0 = 1
|
|
|
|
for i, line in ipairs(Files.get_lines(lua_string)) do
|
|
|
|
_accum_0[_len_0] = ("%3d|%s"):format(i, line)
|
|
|
|
_len_0 = _len_0 + 1
|
|
|
|
end
|
|
|
|
lines = _accum_0
|
|
|
|
end
|
|
|
|
local line_numbered_lua = table.concat(lines, "\n")
|
|
|
|
error("Failed to compile generated code:\n\027[1;34m" .. tostring(line_numbered_lua) .. "\027[0m\n\n" .. tostring(err), 0)
|
|
|
|
end
|
|
|
|
local source_key = tostring(source)
|
|
|
|
if not (environment.SOURCE_MAP[source_key]) then
|
|
|
|
local map = { }
|
|
|
|
local file = Files.read(source.filename)
|
|
|
|
if not file then
|
|
|
|
error("Failed to find file: " .. tostring(source.filename))
|
|
|
|
end
|
|
|
|
local nomsu_str = file:sub(source.start, source.stop)
|
|
|
|
local lua_line = 1
|
|
|
|
local nomsu_line = Files.get_line_number(file, source.start)
|
|
|
|
local map_sources
|
|
|
|
map_sources = 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
|
|
|
|
if s.source and s.source.filename == source.filename then
|
|
|
|
nomsu_line = Files.get_line_number(file, s.source.start)
|
|
|
|
end
|
|
|
|
local _list_0 = s.bits
|
|
|
|
for _index_0 = 1, #_list_0 do
|
|
|
|
local b = _list_0[_index_0]
|
|
|
|
map_sources(b)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
map_sources(to_run)
|
|
|
|
map[lua_line] = map[lua_line] or nomsu_line
|
|
|
|
map[0] = 0
|
|
|
|
environment.SOURCE_MAP[source_key] = map
|
|
|
|
end
|
|
|
|
return run_lua_fn()
|
|
|
|
else
|
|
|
|
return error("Attempt to run unknown thing: " .. tostring(to_run))
|
|
|
|
end
|
|
|
|
end,
|
|
|
|
FILE_CACHE = { },
|
|
|
|
run_file_1_in = function(path, environment, optimization)
|
|
|
|
if optimization == nil then
|
|
|
|
optimization = 1
|
|
|
|
end
|
|
|
|
if environment.FILE_CACHE[path] then
|
|
|
|
import_to_1_from(environment, environment.FILE_CACHE[path])
|
|
|
|
return
|
|
|
|
end
|
2018-11-09 16:40:36 -08:00
|
|
|
if _currently_running_files:has(path) then
|
|
|
|
local i = _currently_running_files:index_of(path)
|
|
|
|
_currently_running_files:add(path)
|
|
|
|
local circle = _currently_running_files:from_1_to(i, -1)
|
|
|
|
print(_currently_running_files, path)
|
|
|
|
error("Circular import detected:\n " .. circle:joined_with("\n..imports "))
|
|
|
|
end
|
|
|
|
_currently_running_files:add(path)
|
2018-11-08 15:23:22 -08:00
|
|
|
local mod = _1_forked(environment)
|
|
|
|
mod._ENV = mod
|
|
|
|
for _, filename in Files.walk(path) do
|
|
|
|
local _continue_0 = false
|
|
|
|
repeat
|
|
|
|
if not (filename == "stdin" or filename:match("%.nom$")) then
|
|
|
|
_continue_0 = true
|
|
|
|
break
|
|
|
|
end
|
|
|
|
local lua_filename = filename:gsub("%.nom$", ".lua")
|
|
|
|
local code
|
|
|
|
if optimization ~= 0 and Files.read(lua_filename) then
|
|
|
|
local file = Files.read(lua_filename)
|
2018-11-09 16:40:36 -08:00
|
|
|
code = LuaCode:from(Source(filename, 1, #file), file)
|
2018-11-08 15:23:22 -08:00
|
|
|
else
|
|
|
|
local file = Files.read(filename)
|
2018-11-09 16:40:36 -08:00
|
|
|
code = NomsuCode:from(Source(filename, 1, #file), file)
|
2018-11-08 15:23:22 -08:00
|
|
|
end
|
|
|
|
environment.run_1_in(code, mod)
|
|
|
|
_continue_0 = true
|
|
|
|
until true
|
|
|
|
if not _continue_0 then
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
|
|
|
import_to_1_from(environment, mod)
|
|
|
|
environment.FILE_CACHE[path] = mod
|
2018-11-09 16:40:36 -08:00
|
|
|
return _currently_running_files:remove()
|
2018-11-08 15:23:22 -08:00
|
|
|
end,
|
|
|
|
compile_error_at = function(tree, err_msg, hint)
|
|
|
|
if hint == nil then
|
|
|
|
hint = nil
|
|
|
|
end
|
|
|
|
local err_str = pretty_error({
|
|
|
|
title = "Compile error",
|
|
|
|
error = err_msg,
|
|
|
|
hint = hint,
|
2018-11-09 14:36:15 -08:00
|
|
|
source = tree:get_source_file(),
|
2018-11-08 15:23:22 -08:00
|
|
|
start = tree.source.start,
|
|
|
|
stop = tree.source.stop,
|
|
|
|
filename = tree.source.filename
|
|
|
|
})
|
|
|
|
return error(err_str, 0)
|
|
|
|
end
|
|
|
|
})
|
|
|
|
nomsu_environment._ENV = nomsu_environment
|
|
|
|
SOURCE_MAP = nomsu_environment.SOURCE_MAP
|
|
|
|
return nomsu_environment
|