332 lines
10 KiB
Lua
332 lines
10 KiB
Lua
local EXIT_SUCCESS, EXIT_FAILURE = 0, 1
|
|
local usage = [=[Nomsu Compiler
|
|
|
|
Usage: (lua nomsu.lua | moon nomsu.moon) [-V version] [-i] [-O] [-v] [-c] [-f] [-s] [--help] [--version] [-p print_file] file1 file2... [-- nomsu args...]
|
|
|
|
OPTIONS
|
|
-i Run the compiler in interactive mode (REPL)
|
|
-O Run the compiler in optimized mode (use precompiled .lua versions of Nomsu files, when available)
|
|
-v Verbose: print compiled lua code
|
|
-c Compile .nom files into .lua files
|
|
-f Auto-format the given Nomsu file and print the result.
|
|
-s Check the program for syntax errors.
|
|
-h/--help Print this message.
|
|
--version Print the version number and exit.
|
|
-V specify which Nomsu version is desired
|
|
-p <file> Print to the specified file instead of stdout.
|
|
<input> Input file can be "-" to use stdin.
|
|
]=]
|
|
local ok, _ = pcall(function()
|
|
lpeg = require('lpeg')
|
|
re = require('re')
|
|
end)
|
|
if not ok then
|
|
print("Error: unable to find the 'lpeg' Lua module. Please install LPEG either from http://www.inf.puc-rio.br/~roberto/lpeg/re.html or, if you use luarocks: `luarocks install lpeg`")
|
|
os.exit(EXIT_FAILURE)
|
|
end
|
|
local Errhand = 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 repr
|
|
repr = require("utils").repr
|
|
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) :}
|
|
/ {:version: ("--version" -> true) :}
|
|
/ {:requested_version: "-V" ((";")? {([0-9.])+})? :}
|
|
file <- "-" / [^;]+
|
|
]], {
|
|
["true"] = function()
|
|
return true
|
|
end
|
|
})
|
|
local arg_string = table.concat(arg, ";") .. ";"
|
|
local args = parser:match(arg_string)
|
|
if not args or args.help then
|
|
print(usage)
|
|
os.exit(EXIT_FAILURE)
|
|
end
|
|
local nomsu = NomsuCompiler
|
|
nomsu.arg = args.nomsu_args
|
|
if args.version then
|
|
nomsu:run('use "core"\nsay (Nomsu version)')
|
|
os.exit(EXIT_SUCCESS)
|
|
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 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 iterate_single
|
|
iterate_single = function(item, prev)
|
|
if item == prev then
|
|
return nil
|
|
else
|
|
return item
|
|
end
|
|
end
|
|
local lfs
|
|
ok, lfs = pcall(require, "lfs")
|
|
if ok then
|
|
all_files = function(path)
|
|
local browse
|
|
browse = function(filename)
|
|
local file_type = lfs.attributes(filename, 'mode')
|
|
if file_type == 'file' then
|
|
if match(filename, "%.nom$") or match(filename, "%.lua$") then
|
|
coroutine.yield(filename)
|
|
return true
|
|
end
|
|
elseif file_type == 'directory' then
|
|
for subfile in lfs.dir(filename) do
|
|
if not (subfile == "." or subfile == "..") then
|
|
browse(filename .. "/" .. subfile)
|
|
end
|
|
end
|
|
return true
|
|
elseif file_type == 'char device' then
|
|
coroutine.yield(filename)
|
|
return true
|
|
end
|
|
return false
|
|
end
|
|
return coroutine.wrap(function()
|
|
if not browse(path) and package.nomsupath then
|
|
browse(package.nomsupath .. "/" .. path)
|
|
end
|
|
return nil
|
|
end)
|
|
end
|
|
else
|
|
local ret = os.execute('find . -maxdepth 0')
|
|
if not (ret == true or ret == 0) then
|
|
error("Could not find 'luafilesystem' module and couldn't run system command `find` (this might happen on Windows). Please install `luafilesystem` (which can be found at: http://keplerproject.github.io/luafilesystem/ or `luarocks install luafilesystem`)", 0)
|
|
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"')
|
|
local found = false
|
|
for line in f:lines() do
|
|
found = true
|
|
coroutine.yield(line)
|
|
end
|
|
if not found and package.nomsupath then
|
|
f:close()
|
|
f = io.popen('find -L "' .. package.nomsupath .. '/' .. path .. '" -not -path "*/\\.*" -type f -name "*.nom"')
|
|
for line in f:lines() do
|
|
coroutine.yield(line)
|
|
end
|
|
end
|
|
local success = f:close()
|
|
if not (success) then
|
|
return error("Invalid file path: " .. tostring(path))
|
|
end
|
|
end)
|
|
end
|
|
end
|
|
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
|
|
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
|
|
nomsu.can_optimize = function(f)
|
|
if not (args.optimized) then
|
|
return false
|
|
end
|
|
if to_run[f] then
|
|
return false
|
|
end
|
|
return true
|
|
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
|
|
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 err
|
|
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
|
|
nomsu:run_file(filename)
|
|
end
|
|
end
|
|
if #parse_errs > 0 then
|
|
io.stderr:write(table.concat(parse_errs, "\n\n"))
|
|
io.stderr:flush()
|
|
os.exit(EXIT_FAILURE)
|
|
elseif args.syntax then
|
|
os.exit(EXIT_SUCCESS)
|
|
end
|
|
if args.interactive then
|
|
nomsu:run('say "\\n\\(bright)\\(underscore)Welcome to the Nomsu v\\(Nomsu version) interactive console!\\(reset color)\\n press \'enter\' twice to run a command\\n"')
|
|
local ready_to_quit = false
|
|
nomsu.A_quit = function()
|
|
ready_to_quit = true
|
|
return print("Goodbye!")
|
|
end
|
|
nomsu.A_exit = nomsu.A_quit
|
|
nomsu.A_help = function()
|
|
print("This is the Nomsu v" .. tostring(nomsu.A_Nomsu_version()) .. " interactive console.")
|
|
print("You can type in Nomsu code here and hit 'enter' twice to run it.")
|
|
return print("To exit, type 'exit' or 'quit' and hit enter twice")
|
|
end
|
|
for repl_line = 1, math.huge do
|
|
io.write(colored.bright(colored.yellow(">> ")))
|
|
local buff = { }
|
|
while true do
|
|
io.write(colors.bright)
|
|
local line = io.read("*L")
|
|
io.write(colors.reset)
|
|
if line == "\n" or not line then
|
|
if #buff > 0 then
|
|
io.write("\027[1A\027[2K")
|
|
end
|
|
break
|
|
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)
|
|
local pseudo_filename = "user input #" .. repl_line
|
|
FILE_CACHE[pseudo_filename] = buff
|
|
local err_hand
|
|
err_hand = function(error_message)
|
|
return Errhand.print_error(error_message)
|
|
end
|
|
local ret
|
|
ok, ret = xpcall(nomsu.run, err_hand, nomsu, buff, Source(pseudo_filename, 1, #buff))
|
|
if ok and ret ~= nil then
|
|
print("= " .. repr(ret))
|
|
elseif not ok then
|
|
Errhand.print_error(ret)
|
|
end
|
|
if ready_to_quit then
|
|
break
|
|
end
|
|
end
|
|
end
|
|
end
|
|
local has_ldt, ldt = pcall(require, 'ldt')
|
|
if has_ldt then
|
|
return ldt.guard(run)
|
|
else
|
|
return Errhand.run_safely(run)
|
|
end
|