nomsu/nomsu.lua

303 lines
8.8 KiB
Lua
Raw Normal View History

if NOMSU_VERSION and NOMSU_PREFIX 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_PREFIX) .. "/share/nomsu/" .. 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_PREFIX) .. "/lib/nomsu/" .. 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_PREFIX) .. "/share/nomsu/" .. 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-07-09 16:58:46 -07:00
Usage: (nomsu | lua nomsu.lua | moon nomsu.moon) [-V version] [-O] [-v] [-c] [-s] [-I file] [--help | -h] [--version] [file [nomsu args...]]
OPTIONS
2018-07-09 16:58:46 -07:00
-O Run the compiler in optimized mode (use precompiled .lua versions of Nomsu files, when available).
-v Verbose: print compiled lua code.
-c Compile the input files into a .lua files.
-s Check the input files for syntax errors.
-I <file> Add an additional input file or directory.
-h/--help Print this message.
2018-06-23 00:57:31 -07:00
--version Print the version number and exit.
2018-07-09 16:58:46 -07:00
-V specify which Nomsu version is desired.
<file> The Nomsu file to run (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
if not arg or debug.getinfo(2).func == require then
return NomsuCompiler
end
2018-07-09 16:58:46 -07:00
local file_queue = { }
local parser = re.compile([[ args <- {| (flag ";")* (({~ file ~} -> add_file) ";")? {:nomsu_args: {| ({[^;]*} ";")* |} :} ";"? |} !.
2018-06-19 01:27:32 -07:00
flag <-
2018-07-09 16:58:46 -07:00
{:optimized: ("-O" -> true) :}
/ ("-I" (";")? ({~ file ~} -> add_file))
/ ({:check_syntax: ("-s" -> true):})
/ ({:compile: ("-c" -> true):})
2018-06-19 01:27:32 -07:00
/ {:verbose: ("-v" -> true) :}
/ {:help: (("-h" / "--help") -> true) :}
2018-06-23 00:57:31 -07:00
/ {:version: ("--version" -> true) :}
/ {:requested_version: "-V" ((";")? {([0-9.])+})? :}
2018-07-09 16:58:46 -07:00
file <- ("-" -> "stdin") / {[^;]+}
2018-06-19 01:27:32 -07:00
]], {
["true"] = function()
return true
2018-07-09 16:58:46 -07:00
end,
add_file = function(f)
return table.insert(file_queue, f)
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()
local input_files = { }
2018-07-09 16:58:46 -07:00
for _index_0 = 1, #file_queue do
local f = file_queue[_index_0]
if not (files.exists(f)) then
error("Could not find: " .. tostring(f))
end
for filename in files.walk(f) do
input_files[filename] = true
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
2018-07-09 16:58:46 -07:00
if args.compile and input_files[f] then
2018-06-23 00:57:31 -07:00
return false
end
return true
end
2018-07-09 16:58:46 -07:00
local get_file_and_source
get_file_and_source = function(filename)
local file, source
if filename == 'stdin' then
file = io.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
return nil
end
source = Source(filename, 1, #file)
return file, source
end
local run_file
run_file = function(filename, lua_handler)
if lua_handler == nil then
lua_handler = nil
end
local file, source = get_file_and_source(filename)
if not (file) then
return nil
end
local tree = nomsu:parse(file, source)
if tree then
if tree.type ~= "FileChunks" then
tree = {
tree
}
end
for _index_0 = 1, #tree do
local chunk = tree[_index_0]
local lua = nomsu:compile(chunk):as_statements("return ")
lua:declare_locals()
lua:prepend("-- File: " .. tostring(source.filename:gsub("\n.*", "...")) .. "\n")
if lua_handler then
lua_handler(tostring(lua))
end
nomsu:run_lua(lua)
end
end
end
2018-06-19 01:27:32 -07:00
local parse_errs = { }
2018-07-09 16:58:46 -07:00
for _index_0 = 1, #file_queue do
local f = file_queue[_index_0]
for filename in files.walk(f) do
local _continue_0 = false
repeat
2018-07-09 16:58:46 -07:00
if args.check_syntax then
local file, source = get_file_and_source(filename)
if not (file) then
_continue_0 = true
break
end
2018-07-09 16:58:46 -07:00
nomsu:parse(file, source)
print("Parse succeeded: " .. tostring(filename))
end
if args.compile then
2018-07-09 16:58:46 -07:00
local output
if filename == 'stdin' then
output = io.output()
else
2018-07-09 16:58:46 -07:00
output = io.open(filename:gsub("%.nom$", ".lua"), "w")
end
run_file(filename, function(lua)
output:write(tostring(lua), "\n")
if args.verbose then
2018-07-09 16:58:46 -07:00
return print(tostring(lua))
end
2018-07-09 16:58:46 -07:00
end)
print(("Compiled %-25s -> %s"):format(filename, filename:gsub("%.nom$", ".lua")))
output:close()
end
2018-07-09 16:58:46 -07:00
if not args.check_syntax and not args.compile then
run_file(filename, (args.verbose and print or nil))
end
_continue_0 = true
until true
if not _continue_0 then
break
2018-06-19 01:27:32 -07:00
end
end
end
2018-07-09 16:58:46 -07:00
if #file_queue == 0 then
2018-06-23 18:26:18 -07:00
nomsu:run([[use "core"
use "lib/consolecolor.nom"
action [quit, exit]: lua> "os.exit(0)"
action [help]
say ".."
This is the Nomsu v\(Nomsu version) interactive console.
You can type in Nomsu code here and hit 'enter' twice to run it.
To exit, type 'exit' or 'quit' and hit enter twice.
2018-06-23 18:26:18 -07:00
say ".."
\(bright)\(underscore)Welcome to the Nomsu v\(Nomsu version) interactive console!\(reset color)
press 'enter' twice to run a command
2018-06-23 18:26:18 -07:00
\("")]])
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
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