nomsu/nomsu.lua

359 lines
11 KiB
Lua
Raw Normal View History

if NOMSU_VERSION and NOMSU_LIB and NOMSU_SHARE then
local ver_bits
do
local _accum_0 = { }
local _len_0 = 1
for ver_bit in NOMSU_VERSION:gmatch("[0-9]+") do
_accum_0[_len_0] = ver_bit
_len_0 = _len_0 + 1
end
ver_bits = _accum_0
end
local partial_vers
do
local _accum_0 = { }
local _len_0 = 1
for i = #ver_bits, 1, -1 do
_accum_0[_len_0] = table.concat(ver_bits, '.', 1, i)
_len_0 = _len_0 + 1
end
partial_vers = _accum_0
end
package.path = table.concat((function()
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1, #partial_vers do
local v = partial_vers[_index_0]
_accum_0[_len_0] = tostring(NOMSU_SHARE) .. "/" .. tostring(v) .. "/?.lua"
_len_0 = _len_0 + 1
end
return _accum_0
end)(), ";") .. ";" .. package.path
package.cpath = table.concat((function()
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1, #partial_vers do
local v = partial_vers[_index_0]
_accum_0[_len_0] = tostring(NOMSU_LIB) .. "/" .. tostring(v) .. "/?.so"
_len_0 = _len_0 + 1
end
return _accum_0
end)(), ";") .. ";" .. package.cpath
package.nomsupath = table.concat((function()
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1, #partial_vers do
local v = partial_vers[_index_0]
_accum_0[_len_0] = tostring(NOMSU_SHARE) .. "/" .. tostring(v)
_len_0 = _len_0 + 1
end
return _accum_0
end)(), ";")
end
local EXIT_SUCCESS, EXIT_FAILURE = 0, 1
2018-06-19 01:27:32 -07:00
local usage = [=[Nomsu Compiler
2018-06-23 00:57:31 -07:00
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.
2018-06-23 00:57:31 -07:00
--version Print the version number and exit.
-V specify which Nomsu version is desired
-p <file> Print to the specified file instead of stdout.
2018-04-28 19:16:39 -07:00
<input> Input file can be "-" to use stdin.
2018-06-19 01:27:32 -07:00
]=]
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
2018-06-19 15:24:24 -07:00
local Errhand = require("error_handling")
2018-06-19 01:27:32 -07:00
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
2018-06-23 00:57:31 -07:00
local repr
repr = require("utils").repr
2018-06-19 01:27:32 -07:00
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) :}
2018-06-23 00:57:31 -07:00
/ {:version: ("--version" -> true) :}
/ {:requested_version: "-V" ((";")? {([0-9.])+})? :}
2018-06-19 01:27:32 -07:00
file <- "-" / [^;]+
]], {
["true"] = function()
return true
end
2018-06-19 01:27:32 -07:00
})
2018-06-23 00:57:31 -07:00
local arg_string = table.concat(arg, ";") .. ";"
local args = parser:match(arg_string)
2018-06-19 01:27:32 -07:00
if not args or args.help then
print(usage)
os.exit(EXIT_FAILURE)
2018-06-19 01:27:32 -07:00
end
local files = require("files")
2018-06-19 01:27:32 -07:00
local nomsu = NomsuCompiler
nomsu.arg = args.nomsu_args
2018-06-23 00:57:31 -07:00
if args.version then
nomsu:run([[use "core"
say (Nomsu version)]])
2018-06-23 00:57:31 -07:00
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
})
2018-06-19 01:27:32 -07:00
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]
local found = false
for f in files.walk(input) do
2018-06-19 01:27:32 -07:00
input_files[#input_files + 1] = f
to_run[f] = true
found = true
end
if not found then
error("Could not find: " .. tostring(input))
2018-06-19 01:27:32 -07:00
end
end
2018-06-23 00:57:31 -07:00
nomsu.can_optimize = function(f)
if not (args.optimized) then
return false
end
if to_run[f] then
return false
end
return true
end
2018-06-19 01:27:32 -07:00
local parse_errs = { }
local _list_1 = args.inputs
for _index_0 = 1, #_list_1 do
local arg = _list_1[_index_0]
for filename in files.walk(arg) do
local _continue_0 = false
repeat
local file, source
if filename == STDIN then
file = io.input():read("*a")
files.spoof('stdin', file)
source = Source('stdin', 1, #file)
elseif filename:match("%.nom$") then
file = files.read(filename)
if not file then
error("File does not exist: " .. tostring(filename), 0)
end
source = Source(filename, 1, #file)
else
_continue_0 = true
break
end
source = Source(filename, 1, #file)
local output
if args.compile then
output = io.open(filename:gsub("%.nom$", ".lua"), "w")
else
output = nil
end
if args.syntax then
local err
ok, err = pcall(nomsu.parse, nomsu, file, source)
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
_continue_0 = true
break
end
local tree = nomsu:parse(file, source)
if args.format then
local formatted = tree and tostring(nomsu:tree_to_nomsu(tree)) or ""
if print_file then
print_file:write(formatted, "\n")
print_file:flush()
end
_continue_0 = true
break
end
if tree then
if tree.type == "FileChunks" then
for _index_1 = 1, #tree do
local chunk = tree[_index_1]
local lua = nomsu:compile(chunk):as_statements("return ")
lua:declare_locals()
lua:prepend("-- File: " .. tostring(source.filename:gsub("\n.*", "...")) .. "\n")
if args.compile then
output:write(tostring(lua), "\n")
end
if args.verbose then
print(tostring(lua))
end
nomsu:run_lua(lua)
end
else
local lua = nomsu:compile(tree):as_statements("return ")
lua:declare_locals()
lua:prepend("-- File: " .. tostring(source.filename:gsub("\n.*", "...")) .. "\n")
if args.compile then
output:write(tostring(lua), "\n")
end
if args.verbose then
print(tostring(lua))
end
nomsu:run_lua(lua)
end
end
if args.compile then
print(("Compiled %-25s -> %s"):format(filename, filename:gsub("%.nom$", ".lua")))
output:close()
end
_continue_0 = true
until true
if not _continue_0 then
break
2018-06-19 01:27:32 -07:00
end
end
end
if #parse_errs > 0 then
io.stderr:write(table.concat(parse_errs, "\n\n"))
io.stderr:flush()
os.exit(EXIT_FAILURE)
2018-06-19 01:27:32 -07:00
elseif args.syntax then
os.exit(EXIT_SUCCESS)
2018-06-19 01:27:32 -07:00
end
if args.interactive then
2018-06-23 18:26:18 -07:00
nomsu:run([[use "core"
use "lib/consolecolor.nom"
say ".."
\(bright)\(underscore)Welcome to the Nomsu v\(Nomsu version) interactive console!\(reset color)
press \'enter\' twice to run a command
\("")]])
2018-06-23 00:57:31 -07:00
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
2018-06-19 01:27:32 -07:00
for repl_line = 1, math.huge do
io.write(colored.bright(colored.yellow(">> ")))
local buff = { }
while true do
2018-06-23 00:57:31 -07:00
io.write(colors.bright)
2018-06-19 01:27:32 -07:00
local line = io.read("*L")
2018-06-23 00:57:31 -07:00
io.write(colors.reset)
2018-06-19 01:27:32 -07:00
if line == "\n" or not line then
if #buff > 0 then
io.write("\027[1A\027[2K")
end
2018-06-19 01:27:32 -07:00
break
end
2018-06-19 01:27:32 -07:00
line = line:gsub("\t", " ")
table.insert(buff, line)
io.write(colored.dim(colored.yellow(".. ")))
end
2018-06-19 01:27:32 -07:00
if #buff == 0 then
break
end
2018-06-19 01:27:32 -07:00
buff = table.concat(buff)
2018-06-19 15:24:24 -07:00
local pseudo_filename = "user input #" .. repl_line
files.spoof(pseudo_filename, buff)
2018-06-19 01:27:32 -07:00
local err_hand
err_hand = function(error_message)
2018-06-19 15:24:24 -07:00
return Errhand.print_error(error_message)
end
local ret
ok, ret = xpcall(nomsu.run, err_hand, nomsu, buff, Source(pseudo_filename, 1, #buff))
2018-06-19 01:27:32 -07:00
if ok and ret ~= nil then
print("= " .. repr(ret))
elseif not ok then
2018-06-19 15:24:24 -07:00
Errhand.print_error(ret)
end
2018-06-23 00:57:31 -07:00
if ready_to_quit then
break
end
end
end
end
2018-06-19 02:00:52 -07:00
local has_ldt, ldt = pcall(require, 'ldt')
if has_ldt then
return ldt.guard(run)
else
2018-06-19 15:24:24 -07:00
return Errhand.run_safely(run)
2018-06-19 02:00:52 -07:00
end